Ricollegalo stasera o questo fine settimana


20

Questo è il mio primo codice golf, quindi per favore fatemi sapere se è troppo ampio o se mi mancano informazioni per un buon puzzle!

Sfida

In Ontario e possibilmente in altre aree del mondo, l'elettricità viene fatturata utilizzando la tariffazione Time-Of-Use (TOU) , che varia il costo per chilowattora in base al consumo di energia.

Data una data e un'ora, voglio sapere se sono in un periodo di picco (rosso), medio (giallo) o non di punta (verde).

Ingresso

Supponiamo che l'input sia fornito in un formato accettabile ISO 8601 data-ora senza fuso orario con la precisione minima delle ore: YYYY-MM-DDThh[:mm[:ss]](la T è letterale).

Esempi

  • 2014-09-01T14
  • 2014-09-01T17: 30
  • 2014-09-01T17: 30: 02

Produzione

L'output dovrebbe essere una stringa On, Mido Off.

Regole

  • Il codice più corto vince
  • Ai fini di questa sfida, ignorare le festività
  • Supponi le informazioni trovate in questo post. Le attuali regole di tariffazione del tempo di utilizzo potrebbero cambiare in futuro dal Ministero dell'Energia dell'Ontario.

Informazione

Giorni feriali estivi (dal 1 maggio al 31 ottobre)

Orologio orario per i giorni feriali estivi

  • Off-peak: 19h00 - 07h00
  • Mid-peak: 07:00 - 11:00 e 17:00 - 19:00
  • Sul picco: 11: 00-17: 00

Giorni feriali invernali (dal 1 novembre al 30 aprile)

Orologio orario per i giorni feriali invernali

  • Off-peak: 19h00 - 07h00
  • Mid-peak: 11h00 - 17h00
  • In vetta: dalle 07:00 alle 11:00 e dalle 17:00 alle 19:00

Fine settimana

Orologio con orario di utilizzo nei fine settimana

  • Off-peak: tutto il giorno

Sei sicuro che i giorni feriali invernali non vengano scambiati di tipo medio-picco / massimo?
John Dvorak,

3
@JanDvorak, in inverno le persone usano le luci e il riscaldamento al mattino e alla sera; in estate usano l'aria condizionata a mezzogiorno.
Peter Taylor,

4
Questo è un duplicato borderline di codegolf.stackexchange.com/q/7008/194 (analizza un datetime e fai un semplice calcolo basato sul fatto che sia un giorno lavorativo o meno). Penso che la dipendenza stagionale sia abbastanza diversa, ma altri potrebbero non essere d'accordo.
Peter Taylor,

@PeterTaylor qui le regole sembrano molto più semplici rispetto alla domanda collegata. Questo non deve gestire anni bisestili, per esempio.
John Dvorak,

3
Il formato della data generale dovrebbe essere YYYY-MM-DDThh[:mm[:ss]]dato che i secondi possono essere applicati solo se vengono applicati i minuti?
Cruncher,

Risposte:


3

Rubino - 147 144 143 141 137 135

x=->s{y,m,d,h=s.scan(/\d+/).map &:to_i
g=Time.new(y,m,d).wday%6<1?0:[0..11,4..9].count{|r|r===h-7}
%W{Off Mid On}[m<5||m>10?(3-g)%3:g]}

Ciò rappresenta una funzione che accetta una stringa come parametro e restituisce una stringa.

Ecco una demo online con alcuni casi di test: http://ideone.com/wyIydw


8

Python 2 - 164

from datetime import*
d,t=input().split('T')
y,m,d=map(int,d.split('-'))
t=int(t[:2])
print'OMOfinfd'[(1+((10<t<17)==(4<m<11)))*(date(y,m,d).weekday()<5<6<t<19)::3]

Se necessario, di seguito è una spiegazione della logica nella riga finale:

La riga finale stampa una porzione di a 'OMOfinfd'seconda della valutazione dei suoi condizionali.

  • Innanzitutto, valuta l'operazione 1+((10<t<17)==(4<m<11)).

    Se l'XNOR tra le condizioni 10<t<17e lo 4<m<11è False, verrà valutato 1+False => 1+0 => 1. Altrimenti, l'operazione valuterà 1+True => 1+1 => 2.

  • Infine, moltiplica quel risultato dell'operazione sopra per se il giorno è un giorno feriale e se l'orario è tra le 6:00 e le 19:00.

    In questo caso False, o il giorno è un fine settimana o l'orario è tra le 19:00 e le 6:00, e il risultato sarà (1|2)*0 => 0. Altrimenti il ​​risultato sarà (1|2)*1 => 1|2.

Un risultato di 0verrà stampato Off, 1verrà stampato Mide 2verrà stampato On.


La versione golfata non è in realtà più lunga perché si usano i punti e virgola? O vengono contate le newline?
Decadimento beta

Le nuove righe vengono in genere conteggiate. In Notepad ++ (dove in genere ottengo i miei conteggi) la versione non controllata è più lunga di 4byte a 168.
BeetDemGuise

3
@BeetDemGuise Modifica → Conversione EOL → Formato UNIX / OSX.
Scisma,

5

C # - 240 220 caratteri

string x(string s){var d=DateTime.Parse((s+":00:00").Substring(0,19));int h=d.Hour,i=(int)d.DayOfWeek,x=d.Month;string o="off",m="mid",f="on";return i==6|i==0?o:x>=5&x<11?h>18|h<7?o:h>10&h<17?f:m:h>18|h<7?o:h>10&h<17?m:f;}

Niente di speciale. Codifica diretta.

Grazie a w0lf :)


1
Penso che puoi accorciare s.Length==13?s+":00:00":s.Length==16?s+":00":sa(s+":00:00").Substring(0,19)
Cristian Lupascu il

5

Rubino - 135

Abusa del modulo Time. Inserito dall'argomento della riga di comando.

d=Time.new(*$*[0].scan(/\d+/)[0..3])
o,m,f=%w{On Mid Off}
o,m=m,o if (d.mon-5)%10<6
p d.wday%6<1||(12>h=(d.hour+5)%24)?f:15<h&&h<22?m:o

Modifica: grazie a w0lf per Time, che ha contribuito ad accorciare e risolvere un bug.


Il tuo programma non è corretto Per l'ingresso 2014-09-01T17:30emette correttamente "Mid", ma per 2014-09-01T17l'uscita "Off".
Cristian Lupascu,

3

Groovy - 621 534 524 491 caratteri

Qualche ulteriore golf da fare, ma piuttosto semplice quando si utilizza Joda-Time

@Grab(group='joda-time',module='joda-time',version='2.3')
f={p,a->org.joda.time.format.DateTimeFormat.forPattern(p).parseDateTime a}
g={p,a->def x=null;try{x=f p,a}catch(Exception e){}}
a=args[0]
d=["",":mm",":mm:ss"].collect{g "yyyy-MM-dd'T'HH$it",a}.find{it}
r="Off"
m="Mid"
j={d,a,b->d.hourOfDay>a&&d.hourOfDay<b}
k={j(it,6,11)||(j(it,16,19))}
if(d.dayOfWeek<6){x=d.monthOfYear;if(x>4&&x<12){if(j(d,10,17))r="On";if(k(d))r=m}else if(x<5||x>10){if(j(d,10,17))r=m;if(k(d))r="On"}}
println r

esecuzioni campione:

bash-3.2$ ./run.peak.sh 
groovy Peak.groovy 2014-08-26T19
Off
groovy Peak.groovy 2014-08-26T07:00
Mid
groovy Peak.groovy 2014-08-26T18:00:00
Mid
groovy Peak.groovy 2014-08-26T12:30:30
On
groovy Peak.groovy 2014-11-01T00
Off
groovy Peak.groovy 2014-02-05T11:11:11
Mid
groovy Peak.groovy 2014-01-05T08:08
Off
groovy Peak.groovy 2014-12-18T18:59:59
On
groovy Peak.groovy 2014-08-31T14
Off

Ungolfed:

@Grab(group='joda-time',module='joda-time',version='2.3')

f = { p,a -> org.joda.time.format.DateTimeFormat.forPattern(p).parseDateTime a}
g = { p,a -> def x=null; try{x=f p,a} catch(Exception e) {} }
a=args[0]

d = ["",":mm",":mm:ss"].collect{g "yyyy-MM-dd'T'HH$it",a}.find{it}
r = "Off"
m = "Mid"

j = {d,a,b -> d.hourOfDay > a && d.hourOfDay < b}
k = { j(it,6,11) || (j(it,16,19)) }

if (d.dayOfWeek<6) {
    x = d.monthOfYear;
    if ( x>4 && x<12 ) {
        if (j(d,10,17)) r="On";
        if (k(d)) r=m
    } else if (x<5||x>10) {
        if (j(d,10,17)) r=m;
        if (k(d)) r="On"
    }
}
println r

2
Da Java 8, c'è anche java.time.*, che è molto simile a Joda Time ma fa parte di JRE. Forse questo potrebbe abbreviare un po 'il codice.
ntoskrnl,

ri: Java 8. ottimo punto
Michael Easter,

Nel tuo terzo penultimo esempio '2014-01-05T08:08', il 5 gennaio 2014 è una domenica. Quindi, dovrebbe essere'Off'
BeetDemGuise,

ri: 5 gennaio 2014. Esatto. Il codice corrente è corretto, ma l'esecuzione dell'output è errata. Risolverà.
Michael Easter,

Provato java.time.*. fallito miseramente. Il parser è troppo dannatamente rigoroso. Usa DateParserda nashorn (anche parte di JRE8) che è indulgente, e con un po 'di hacking abusivo, anche abbastanza indulgente da saltare minuti e secondi.
Mark Jeronimus,

2

R, 243 204 caratteri

b=strptime(scan(,""),"%Y-%m-%dT%H");i=function(x)as.integer(format(b,x));h=i("%H");d=i("%m%d");w=d>1100|d<430;f=ifelse;cat(c("Off","Mid","On")[f(i("%u")%in%5:6|h<7|h>19,1,f(h>11&h<17,f(w,2,3),f(w,3,2)))])

Rientrato e commentato:

b=strptime(scan(,""),"%Y-%m-%dT%H") #Takes stdin and converts into POSIXct
i=function(x)as.integer(format(b,x)) #Format the POSIXct and convert it to integer
h=i("%H")      #Format to hours
d=i("%m%d")    #Format to Month/Day
w=d>1100|d<430 #True if winter time, false if summer
f=ifelse
cat(c("Off","Mid","On")[f(i("%u")%in%5:6|h<7|h>19, #If weekend or night
                          1,                       #Case 1
                          f(h>11&h<17,            #Else if mid-day
                             f(w,3,2),             #Case 2 in winter, case 3 in summer
                             f(w,2,3)))])          #else vice versa

Esempi:

> b=strptime(scan(,""),"%Y-%m-%dT%H");i=function(x)as.integer(format(b,x));h=i("%H");d=i("%m%d");w=d>1100|d<430;f=ifelse;cat(c("Off","Mid","On")[f(i("%u")%in%5:6|h<7|h>19,1,f(h>11&h<17,f(w,3,2),f(w,2,3)))])
1: 2014-08-26T15
2: 
Read 1 item
On
> b=strptime(scan(,""),"%Y-%m-%dT%H");i=function(x)as.integer(format(b,x));h=i("%H");d=i("%m%d");w=d>1100|d<430;f=ifelse;cat(c("Off","Mid","On")[f(i("%u")%in%5:6|h<7|h>19,1,f(h>11&h<17,f(w,3,2),f(w,2,3)))])
1: 2014-12-10T15
2: 
Read 1 item
Mid
> b=strptime(scan(,""),"%Y-%m-%dT%H");i=function(x)as.integer(format(b,x));h=i("%H");d=i("%m%d");w=d>1100|d<430;f=ifelse;cat(c("Off","Mid","On")[f(i("%u")%in%5:6|h<7|h>19,1,f(h>11&h<17,f(w,3,2),f(w,2,3)))])
1: 2014-08-26T23
2: 
Read 1 item
Off

1

Bash, 286

questa è una semplice risposta bash usando il programma data

d(){ date -d $1 +%$2; };D=$(echo $1|sed 's/\(T..\)$/\1:00/');H=$(d $D H);M=$(d $D m);if [ $(d $D u) -gt 5 ]||[ $H -lt 7 ]||[ $H -gt 18 ];then echo Off;exit;fi;if [ $M -gt 4 ]&&[ $M -lt 11 ];then I=On;O=Mid;else I=Mid;O=On;fi;if [ $H -gt 10 ]&&[ $H -lt 17 ];then echo $I;else echo $O;fi

1

Eccone un altro!

JavaScript, 175 171

function f(x){d=new Date(x.slice(0,10));t=x.slice(11,13),m=(x.slice(5,7)+1)%12;return(t<8||t>18||!(d.getUTCDay()%6)?'off':((t<11||t>17)?(m<5?'on':'mid'):(m<5?'mid':'on'))}

Unminified:

function f(x) {
  d = new Date(x.slice(0, 10));
  t = x.slice(11, 13), m = (x.slice(5, 7) + 1) % 12;
  return (t < 8 || t > 18 || !(d.getUTCDay() % 6) ? 'off' : ((t < 11 || t > 17) ? (m < 5 ? 'on' : 'mid') : (m < 5 ? 'mid' : 'on'))
}

Funziona solo su interpreti in cui una stringa di data ISO8601 può essere passata al Datecostruttore.

CoffeeScript, 192 189

Sorprendentemente, è più lungo in CoffeeScript perché non esiste un operatore ternario in quella lingua (che come puoi vedere dal mio JavaScript, su cui mi sono fortemente affidato).

f=(x)->d=new Date(x.slice(0,10));t=x.slice(11,13);m=(x.slice(5,7)+1)%12;return'off'if(t<8||t>18||!(d.getUTCDay()%6));if(t<11||t>17)then(if m<5then'on'else'mid')else(if m<5then'mid'else'on')

0

ES6 - 146

Questo è in forma di funzione, usa un paio di brutti hack.

let y,m,d,h,c=s=>[y,m,d,h]=s.split(/[T:-]/g).map(i=>+i),new Date(y,m,d).getDay()in{0:1,6:1}||h<20||8>h?'Off':['Mid','On'][(10>h&&h>16)^(m>5||m<8)]

Ha spiegato:

// These variables are declared outside of the function scope to save
// characters.
let y, // year
    m, // month
    d, // day
    h, // hour
    c = s => // c for check, called as c(isoString)
      [y, m, d, h] = // Assign to fields
        s.split(/[T:-]/g) // Split at delimiters
         .map(i => +i), // Convert all to numbers
                        // Comma used to keep it as a single statement.
      new Date(y, m, d).getDay() // Create Date to get day of week
        in {0:1, 6:1} // Check if it is good (0 = Sunday, 6 = Saturday). It
                      // is shorter to use an object literal than to
                      // do individual checks.
      ||
      h < 20 || 8 > h ? // h is between 7 and 19 (7pm)
       'Off' : // Return 'Off'
       ['Mid','On'][ // Two other outputs
        (10 > h && h > 16) ^ // Check if it is 11-16 (5pm)
        (m > 5 || m < 8)] // Invert that if the month is in summer/fall. Note
                          // that this returns a number, either 1 or 0. This
                          // is an ugly hack using the bitwise xor operator.

0

Python 3 - 352 caratteri

import datetime as dt
t=input()
i=int
p=print
a='on'
b='mid'
c='off'
m=i(t[5:7])
h=i(t[11:13])
d=dt.date(i(t[0:4]),m,i(t[8:10])).weekday()
if 5==d or 6==d:p(c)
elif h>=19 and h<7:p(c)
elif m<=10 and m>=4:
 if h>=7 and h<11:p(b)
 if h>=11 and h<17:p(a)
 if h>=17 and h<19:p(b)
else:
 if h>=7 and h<11:p(a)
 if h>=11 and h<17:p(b)
 if h>=17 and h<19:p(a)

1
Dovresti cambiare s=['high','mid','off']in s=['on','mid','off']- non solo questo salva 2 caratteri, ma la specifica dice di produrre "on".
Sellyme,

0

Java - 426 309/301? (vedi commenti)

String z(String a){
    DateParser b=new DateParser(a);
    boolean c=b.parseEcmaDate();
    Integer[]d=b.getDateFields();
    GregorianCalendar e=new GregorianCalendar(d[0],d[1]-(c?0:1),d[2]);
    e.setTimeZone(new SimpleTimeZone(0,""));
    return(124>>e.get(7)&1)>0&d[3]>6&d[3]<19?
           d[3]>10&d[3]<17^(1008>>e.get(2)&1)>0?"Mid":"On":"Off";
}

Esempio di output:

2014-03-02T00   Off
2014-03-02T06   Off
2014-03-02T07   Off
2014-03-02T10   Off
2014-03-02T11   Off
2014-03-02T16   Off
2014-03-02T17   Off
2014-03-02T18   Off
2014-03-02T19   Off
2014-03-02T23   Off
2014-04-02T00   Off
2014-04-02T06   Off
2014-04-02T07   On
2014-04-02T10   On
2014-04-02T11   Mid
2014-04-02T16   Mid
2014-04-02T17   On
2014-04-02T18   On
2014-04-02T19   Off
2014-04-02T23   Off
2014-05-02T00   Off
2014-05-02T06   Off
2014-05-02T07   Mid
2014-05-02T10   Mid
2014-05-02T11   On
2014-05-02T16   On
2014-05-02T17   Mid
2014-05-02T18   Mid
2014-05-02T19   Off
2014-05-02T23   Off

Ho usato lo stesso trucco EXOR dell'invio di Python. Ho anche usato una +funzione OR, per quando lo è weekend OR night.

L'altro mio grande trucco: maschere di bit.

Ad esempio, per vedere se un numero è compreso tra 2 e 6 (dal lunedì al venerdì), innanzitutto creare un modello di bit in cui i valori interessanti sono 1:

  6   2 
0b1111100 = 124

Quindi, utilizzare bit shift per ottenere il bit interessante sull'LSB ed estrarlo:

(124 >> day_of_week) & 1

Allo stesso modo, ho creato modelli di bit per mesi e ore:

  9    4
0b1111110000 = 1008

      76          76
0b111110000000000001111111 = 16253055
0b000000011111100000000000 = 129024

Sfortunatamente, x>y&x<znella maggior parte dei casi è semplicemente più breve, quindi non l'ho usato in alcuni punti.

E infine, un po 'di pirateria informatica (fortemente dipendente dall'implementazione) con jdk.nashorn.internal.parser.DateParser: Quando parseEcmaDate()non analizza completamente una data (come quando legge l'ora e raggiunge la fine della stringa), restituisce false.

  • Quando termina correttamente, la data viene memorizzata in un Integer[](ftw auto-unboxing), con il mese fissato su base-0 (come altre classi Java).
  • Quando restituisce false, si interrompe a metà strada e non esegue questa correzione. Tuttavia, inserisce comunque qualsiasi cosa analizzata nell'array, che è prontamente disponibile. Da qui il -(c?0:1).

Penso che usare .parse()anche potrebbe funzionare (radendo 8 caratteri), ma non l'ho testato a fondo con input diversi. parsechiama internamente parseEcmaDatee, se fallisce, chiama parseLegacyDate. Quest'ultimo potrebbe rovinare l'array, ma ciò non è accaduto con alcuni casi che ho testato.
Mark Jeronimus,

-1

Niente mi impedisce di entrare nel mio, e ce ne sono comunque altri più corti qui, quindi ...

PHP 5.4+, 194

<?function f($x){$t=substr($x,11,2);$m=(substr($x,5,2)+1)%12;if($t<8||$t>18||(new DateTime(substr($x,0,10)))->format('N')>5)return'off';return($t<11||$t>17)?($m<5?'on':'mid'):($m<5?'mid':'on');}

Nonminified e commentato:

<?
function f($x) {
  $t = substr($x,11,2); // hour
  $m = (substr($x,5,2) + 1) % 12; // month shifted up by 1

  if ($t < 8 || $t > 18 || (new DateTime(substr($x,0,10)))->format('N') > 5)
    return 'off'; // evenings and weekends

  return ($t < 11 || $t > 17)
    ? ($m < 5 ? 'on' : 'mid') // morning and mid-afternoon
    : ($m < 5 ? 'mid' : 'on'); // afternoon
}

Si noti inoltre che è date.timezonenecessario impostare la direttiva in php.ini, altrimenti verrà generata un'eccezione.

Utilizzando il nostro sito, riconosci di aver letto e compreso le nostre Informativa sui cookie e Informativa sulla privacy.
Licensed under cc by-sa 3.0 with attribution required.