Podłącz ponownie dziś wieczorem lub w ten weekend

20

To jest mój pierwszy kod golfowy, więc daj mi znać, jeśli jest zbyt szeroki lub jeśli brakuje mi informacji na temat dobrej układanki!

Wyzwanie

W Ontario i prawdopodobnie w innych częściach świata za energię elektryczną naliczane są ceny według czasu użytkowania (TOU) , który zmienia koszt za kilowatogodzinę w zależności od tego, kiedy używasz energii.

Biorąc pod uwagę datę i godzinę, chcę wiedzieć, czy jestem w szczycie (czerwony), w środku szczytu (żółty), czy poza szczytem (zielony).

Wejście

Załóżmy, że dane wejściowe są dostarczane w akceptowalnym formacie daty i godziny ISO 8601 bez strefy czasowej z minimalną dokładnością godzin: YYYY-MM-DDThh[:mm[:ss]](T jest dosłowne).

Przykłady

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

Wynik

Wyjście powinno być ciągiem On, Midalbo Off.

Zasady

  • Najkrótszy kod wygrywa
  • Na potrzeby tego wyzwania należy zignorować dni ustawowo wolne od pracy
  • Załóżmy, że informacje zawarte w tym poście. Rzeczywiste zasady ustalania cen za czas użytkowania mogą ulec zmianie w przyszłości przez Ministerstwo Energii Ontario.

Informacja

Letnie dni powszednie (od 1 maja do 31 października)

Zegar czasu użytkowania w letnie dni powszednie

  • Poza szczytem: 19.00 - 07.00
  • Środkowy szczyt: 07.00 - 11.00 i 17.00 - 19.00
  • Na szczycie: 11.00 - 17.00

Zimowe dni powszednie (od 1 listopada do 30 kwietnia)

Zegar czasu użytkowania w zimowe dni powszednie

  • Poza szczytem: 19.00 - 07.00
  • Środkowy szczyt: 11.00 - 17.00
  • Na szczycie: 07.00 - 11.00 i 17.00 - 19.00

Weekendy

Zegar czasu użytkowania w weekendy

  • Poza szczytem: Cały dzień
lodowisko. dozorca 6
źródło
Czy jesteś pewien, że w zimowe dni powszednie nie zamieniono środkowych szczytów / szczytów?
John Dvorak
3
@JanDvorak, zimą ludzie używają światła i ogrzewania rano i wieczorem; latem używają klimatyzacji w południe.
Peter Taylor
4
Jest to kopia graniczna codegolf.stackexchange.com/q/7008/194 ( parsuj datę i wykonaj proste obliczenia na podstawie tego, czy jest to dzień roboczy, czy nie). Myślę, że zależność od sezonu jest po prostu inna, ale inni mogą się nie zgadzać.
Peter Taylor
@PeterTaylor zasady wydają się tutaj znacznie prostsze niż w powiązanym pytaniu. Nie musi to na przykład obejmować lat przestępnych.
John Dvorak
3
Czy ogólny format daty powinien być taki, YYYY-MM-DDThh[:mm[:ss]]że sekundy można zastosować tylko wtedy, gdy stosuje się minuty?
Cruncher

Odpowiedzi:

3

Rubinowy - 147 144 143 141 137 135 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]}

To reprezentuje funkcję, która przyjmuje ciąg znaków jako parametr i zwraca ciąg znaków.

Oto demo online z niektórymi przypadkami testowymi: http://ideone.com/wyIydw

Cristian Lupascu
źródło
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]

W razie potrzeby poniżej znajduje się wyjaśnienie logiki w ostatnim wierszu:

Ostatni wiersz wypisuje wycinek w 'OMOfinfd'zależności od oceny jego warunków warunkowych.

  • Najpierw oceń operację 1+((10<t<17)==(4<m<11)).

    Jeśli XNOR między warunkami 10<t<17a 4<m<11jest False, to oceni to na 1+False => 1+0 => 1. W przeciwnym razie operacja zostanie oszacowana na 1+True => 1+1 => 2.

  • Na koniec pomnóż wynik powyższej operacji przez to, czy dzień jest dniem tygodnia i czy czas jest między 6 rano a 19:00.

    Jeśli tak jest False, albo dzień jest weekendem, albo czas jest między 19:00 a 6:00, a wynik będzie (1|2)*0 => 0. W przeciwnym razie wynik będzie (1|2)*1 => 1|2.

Wynik 0zostanie wydrukowany Off, 1wydrukowany Midi 2wydrukowany On.

BeetDemGuise
źródło
Czy wersja w golfa nie jest już dłuższa, ponieważ używasz średników? Czy liczone są nowe linie?
Beta Decay
Nowe linie są zwykle liczone. W Notepad ++ (gdzie zwykle się mylę) wersja bez golfa jest o 4 bajty dłuższa o 168.
BeetDemGuise
3
@BeetDemGuise Edycja → Konwersja EOL → Format UNIX / OSX.
Schizm
5

C # - 240 220 znaków

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;}

Nic specjalnego. Proste kodowanie.

Dzięki w0lf :)

Stephan Schinkel
źródło
1
Myślę, że można skrócić s.Length==13?s+":00:00":s.Length==16?s+":00":sdo(s+":00:00").Substring(0,19)
Cristian Lupascu
5

Rubin - 135

Nadużywa moduł czasu. Wprowadź za pomocą argumentu wiersza poleceń.

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

Edycja: Dzięki w0lf za czas, który pomógł skrócić i rozwiązać błąd.

Vectorized
źródło
Twój program jest nieprawidłowy. Dla wejścia 2014-09-01T17:30poprawnie wyprowadza "Mid", ale dla 2014-09-01T17wyjścia "Off".
Cristian Lupascu,
3

Groovy - 621 534 524 491 znaków

Trochę gry w golfa do zrobienia, ale całkiem proste, jeśli wykorzystasz czas Joda

@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

przykładowe przebiegi:

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

Nie golfowany:

@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
Michael Easter
źródło
2
Od wersji Java 8 jest też java.time.*bardzo podobna do Joda Time, ale część JRE. Być może może to nieco skrócić kod.
ntoskrnl
Odp .: Java 8. doskonały punkt
Michael Easter
W Twoim ostatnim przykładzie '2014-01-05T08:08', 5 stycznia 2014 r. To niedziela. Tak więc powinno być'Off'
BeetDemGuise
Odp .: 5 stycznia 2014. Całkiem słusznie. Aktualny kod jest poprawny, ale przebieg wyjściowy jest nieprawidłowy. Naprawię.
Michał Wielkanoc
Próbowałem java.time.*. zawiodło nieszczęśliwie. Parser jest zbyt cholernie surowy. Używaj DateParserz nashorn (również części JRE8), który jest łagodny i z odrobiną obraźliwego hakowania, nawet wystarczająco łagodny, aby pominąć minuty i sekundy.
Mark Jeronimus
2

R, 243 204 znaków

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)))])

Wcięte i skomentowane:

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

Przykłady:

> 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
plannapus
źródło
1

Bash, 286

jest to prosta odpowiedź bash przy użyciu programu daty

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
Optokopper
źródło
1

Oto kolejna!

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'))
}

Działa tylko w przypadku interpreterów, w których ciąg daty ISO8601 można przekazać do Datekonstruktora.

CoffeeScript, 192 189

Zaskakujące jest to, że jest już w CoffeeScript, ponieważ nie ma trójskładnikowego operatora w tym języku (co, jak widać z mojego JavaScript, polegałem na nim).

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')
lodowisko. dozorca 6
źródło
0

ES6 - 146

To jest w formie funkcji, wykorzystuje kilka paskudnych hacków.

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)]

Wyjaśniono:

// 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.
Isiah Meadows
źródło
0

Python 3 - 352 znaków

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)
Rozpad beta
źródło
1
Powinieneś zmienić s=['high','mid','off']na s=['on','mid','off']- nie tylko oszczędza to 2 znaki, ale specyfikacja mówi, aby wypisywać „on”.
Sellyme
0

Java - 426 309/301? (Zobacz komentarze)

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";
}

Przykładowe dane wyjściowe:

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

Użyłem tej samej sztuczki EXOR, co przesłanie Pythona. Użyłem również +jako funkcji OR, kiedy to jest weekend OR night.

Moja druga wielka sztuczka: maski bitowe.

Na przykład, aby sprawdzić, czy liczba zawiera się między 2 a 6 (od poniedziałku do piątku), najpierw utwórz wzór bitowy, w którym interesujące wartości to 1:

  6   2 
0b1111100 = 124

Następnie użyj przesunięcia bitów, aby uzyskać interesujący bit do LSB i wyodrębnić go:

(124 >> day_of_week) & 1

Podobnie zrobiłem wzory bitów dla miesięcy i godzin:

  9    4
0b1111110000 = 1008

      76          76
0b111110000000000001111111 = 16253055
0b000000011111100000000000 = 129024

Niestety okazuje się, że x>y&x<zw większości przypadków jest po prostu krótszy, więc nie korzystałem z niego w niektórych miejscach.

I na koniec trochę hakowania (w dużym stopniu zależne od implementacji) z jdk.nashorn.internal.parser.DateParser: Kiedy parseEcmaDate()nie analizuje całkowicie daty (na przykład, gdy czyta godzinę i osiąga koniec łańcucha), zwraca false.

  • Po prawidłowym zakończeniu data jest zapisywana w Integer[](automatyczne rozpakowywanie plików ftw), a miesiąc jest ustawiony na 0 (podobnie jak inne klasy Java).
  • Kiedy zwraca false, przerywa się w połowie i nie naprawia tego. Jednak nadal umieszcza wszystko, co analizuje, w tablicy, która jest łatwo dostępna. Stąd -(c?0:1).
Mark Jeronimus
źródło
Myślę, że używanie .parse()może również działać (golić 8 znaków), ale nie przetestowałem tego dokładnie przy różnych wejściach. parsepołączenia wewnętrzne, parseEcmaDatea jeśli się nie powiedzie, połączenia parseLegacyDate. Ten ostatni może zepsuć tablicę, ale nie zdarzyło się to w kilku testowanych przypadkach.
Mark Jeronimus
-1

Nic nie powstrzymuje mnie przed wejściem do własnego, a i tak są tu inne krótsze, więc…

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');}

Unminified and commented:

<?
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
}

Zauważ też, że date.timezonedyrektywa w php.ini musi być ustawiona, w przeciwnym razie zostanie zgłoszony wyjątek.

lodowisko. dozorca 6
źródło