Подключите его сегодня вечером или в эти выходные


20

Это мой первый гольф-код, поэтому, пожалуйста, дайте мне знать, если он слишком широкий или мне не хватает информации для хорошей головоломки!

Вызов

В Онтарио и, возможно, в других регионах мира счета за электроэнергию выставляются по времени использования (TOU) , которое зависит от стоимости киловатт-часа в зависимости от того, когда вы используете электроэнергию.

Учитывая дату и время, я хочу знать, нахожусь ли я в пиковый (красный), средний пиковый (желтый) или непиковый (зеленый) период времени.

вход

Предположим, что ввод осуществляется в приемлемом формате даты-времени ISO 8601 без часового пояса с минимальной точностью часов: YYYY-MM-DDThh[:mm[:ss]](T буквально).

Примеры

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

Выход

Вывод должен быть строкой On, Midили Off.

правила

  • Самый короткий код выигрывает
  • Для целей этого вызова игнорировать установленные законом праздники
  • Предположим, информация найдена в этом посте. Фактические правила ценообразования по времени использования могут измениться в будущем Министерством энергетики Онтарио.

Информация

Летние будни (с 1 мая по 31 октября)

Часы времени использования для летних будней

  • Вне пик: с 19:00 до 07:00
  • Середина пика: с 07:00 до 11:00 и с 17:00 до 19:00
  • На пике: 11:00 - 17:00

Зимние будни (с 1 ноября по 30 апреля)

Часы времени использования для зимних будней

  • Вне пик: с 19:00 до 07:00
  • Середина пика: 11:00 - 17:00
  • В часы пик: с 07:00 до 11:00 и с 17:00 до 19:00

Выходные дни

Время использования часов по выходным

  • Вне пик: весь день

Вы уверены, что зимние будни не поменялись местами?
Джон Дворак

3
@JanDvorak, зимой люди используют свет и отопление утром и вечером; летом они используют кондиционер в полдень.
Питер Тейлор

4
Это дублирующий границу codegolf.stackexchange.com/q/7008/194 (проанализируйте дату и сделайте простой расчет, основываясь на том, рабочий это день или нет). Я думаю, что сезонная зависимость достаточно различна, но другие могут не согласиться.
Питер Тейлор

@PeterTaylor правила кажутся здесь намного проще, чем в связанном вопросе. Это не должно обрабатывать високосные годы, например.
Джон Дворак

3
Должен ли общий формат даты быть, YYYY-MM-DDThh[:mm[:ss]]поскольку секунды могут быть применены, только если применяются минуты?
Cruncher

Ответы:


3

Рубин - 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]}

Это представляет функцию, которая принимает строку в качестве параметра и возвращает строку.

Вот онлайн-демонстрация с некоторыми тестами: 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]

При необходимости ниже приводится объяснение логики в последней строке:

В последней строке печатается срез в 'OMOfinfd'зависимости от оценки его условных выражений.

  • Сначала оцените операцию 1+((10<t<17)==(4<m<11)).

    Если XNOR между условиями 10<t<17и 4<m<11есть False, это будет оцениваться до 1+False => 1+0 => 1. В противном случае операция будет оценена как 1+True => 1+1 => 2.

  • Наконец, умножьте этот результат вышеупомянутой операции на то, является ли день рабочим днем, и является ли время между 6:00 и 19:00.

    Если это так False, то либо выходной, либо время между 19:00 и 6:00, и результат будет (1|2)*0 => 0. В противном случае результат будет (1|2)*1 => 1|2.

Результат 0будет распечатан Off, 1распечатан Midи 2распечатан On.


Разве гольф-версия на самом деле не длиннее, потому что вы используете точки с запятой? Или переводятся строки?
Бета-распад

Новые строки обычно подсчитываются. В Notepad ++ (где я обычно получаю свои подсчеты) версия без гольфа на 4 байта длиннее в 168.
BeetDemGuise

3
@BeetDemGuise Редактировать → Преобразование EOL → Формат UNIX / OSX.
Раскол

5

C # - 240 220 символов

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

Ничего особенного. Прямое кодирование.

Спасибо w0lf :)


1
Я думаю , что вы можете сократить s.Length==13?s+":00:00":s.Length==16?s+":00":sдо(s+":00:00").Substring(0,19)
Cristian Lupaşcu

5

Рубин - 135

Злоупотребляет модулем времени. Ввод аргументом командной строки.

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

Изменить: Спасибо w0lf за время, которое помогло сократить и решить ошибку.


Ваша программа неверна. Для входа 2014-09-01T17:30он правильно выводит "Mid", а для 2014-09-01T17него выводит "Off".
Кристиан Лупаску,

3

Groovy - 621 534 524 491 символ

Еще немного игры в гольф, но довольно просто, используя 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

пробные прогоны:

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
Начиная с Java 8, есть также java.time.*, которая очень похожа на Joda Time, но является частью JRE. Возможно, это может немного сократить код.
ntoskrnl

Re: Java 8. Отличный момент
Майкл Пасха

В вашем последнем примере '2014-01-05T08:08'5 января 2014 года - воскресенье. Таким образом, это должно быть'Off'
BeetDemGuise

Re: 5 января 2014 года. Совершенно верно. Текущий код верен, но выходной прогон неправильный. Починю.
Майкл Пасха

Пробовал java.time.*. с треском провалился. Парсер слишком чертовски строг. Используйте DateParserиз Nashorn (также часть JRE8), который снисходительный, и с небольшим количеством оскорбительного хакерства, даже достаточно снисходительный, чтобы пропустить минуты и секунды.
Марк Жеронимус

2

R, 243 204 символа

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

Отступы и комментарии:

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

Примеры:

> 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

Баш, 286

это простой ответ bash с использованием программы date

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

Здесь идет еще один!

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

Работает только на интерпретаторах, где строка даты ISO8601 может быть передана в Dateконструктор.

CoffeeScript, 192 189

Удивительно, но он больше в CoffeeScript, потому что в этом языке нет троичного оператора (на который я, как вы можете видеть из моего JavaScript, я очень полагался).

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

Это в функциональной форме, использует пару неприятных хаков.

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

Разъяснение:

// 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 символа

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
Вы должны изменить s=['high','mid','off']на s=['on','mid','off']- не только это сохраняет 2 символа, но и спецификация говорит, что для вывода нужно включить.
Sellyme

0

Ява - 426 309/301? (см. комментарии)

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

Пример вывода:

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

Я использовал тот же трюк EXOR, что и представление Python. Я также использовал +в качестве функции ИЛИ, когда это weekend OR night.

Мой другой большой трюк: битовые маски.

Например, чтобы увидеть, находится ли число от 2 до 6 (с понедельника по пятницу), сначала создайте битовую комбинацию, в которой интересные значения равны 1:

  6   2 
0b1111100 = 124

Затем используйте битовое смещение, чтобы получить интересующий бит в LSB и извлечь его:

(124 >> day_of_week) & 1

Точно так же я делал битовые паттерны месяцами и часами:

  9    4
0b1111110000 = 1008

      76          76
0b111110000000000001111111 = 16253055
0b000000011111100000000000 = 129024

К сожалению, x>y&x<zв большинстве случаев получается просто короче, поэтому я не использовал его в некоторых местах.

И, наконец, немного хакерства (сильно зависит от реализации) с jdk.nashorn.internal.parser.DateParser: Когда parseEcmaDate()не полностью анализируется дата (например, когда она читает час и достигает конца строки), она возвращает false.

  • Когда он заканчивается должным образом, дата сохраняется в Integer[](автоматическом распаковывании ftw) с фиксированным месяцем равным base-0 (как и другие классы Java).
  • Когда он возвращает false, он прерывается на полпути и не делает этого исправления. Тем не менее, он по-прежнему помещает все, что он анализировал в массив, который легко доступен. Отсюда -(c?0:1).

Я думаю, что использование just .parse()могло бы также сработать (сбрить 8 символов), но я не проверил это тщательно с разными входами. parseвнутренне звонки parseEcmaDateи, если это не удается, звонки parseLegacyDate. Последний может испортить массив, но этого не произошло с несколькими случаями, которые я тестировал.
Марк Жеронимус

-1

Ничто не мешает мне войти в свой собственный, и в любом случае здесь есть и другие, более короткие, так что ...

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 и прокомментировал:

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

Также обратите внимание, что date.timezoneдиректива в php.ini должна быть установлена, в противном случае будет сгенерировано исключение.

Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.