Праздничная головная боль

14

Предупреждение: НЕ пользуйтесь медицинскими советами из этого поста. Если вам нужна медицинская консультация, обратитесь к квалифицированному специалисту.

У меня болит голова. Мне нужны таблетки от головной боли.

Я расскажу вам о последних нескольких дозах, и вы скажете мне, когда я смогу принять следующую дозу без передозировки.

Я дам вам эту строку: P: 00:00, I: 02:00, P: 04:00, I: 06:00

И ты дашь мне это: Next P: 08:00, I: 10:00

Входные данные:

Строка, представляющая время приема каждого лекарства, в следующем формате:

P: 00:00, I: 02:00, P: 04:00, I: 06:00

Это означает, что Парацетамол был взят в 00:00 и 04:00, а Ибупрофен был принят в 02:00 и 06:00.

Выход (обновлено):

Строка с указанием времени, в течение которого можно принимать следующие лекарства, в следующем формате:

Next P: 08:00, I: 10:00
  • Порядок вывода должен соответствовать порядку приема лекарства. - Если ибупрофен следует принимать в 09:35, а парацетамол - в 10:22, то результат должен бытьNext I: 09:35, P: 10:22
  • Если время для следующей дозы каждого лекарства одинаково, порядок вывода не имеет значения: Next P: 08:00, I: 08:00ИЛИNext I: 08:00, P: 08:00
  • Если принимается только одно лекарство (во входной строке), то только выходное лекарство должно быть в выходной строке: Next P: 02:00

Правила:

  • Будет только два вида лекарств, парацетамол «P» и ибупрофен «I».
  • Парацетамол можно принимать один раз каждые 4 часа, максимум 4 раза в течение 24 часов.
  • Ибупрофен также можно принимать один раз каждые 4 часа, максимум 4 раза в течение 24 часов.
  • Парацетамол и ибупрофен можно принимать вместе или в разное время. Один не учитывается в дозировке другого.
  • Время во входной строке всегда будет последовательным, но может быть больше полуночи (23:00 -> 03:00).
  • Время во входной строке не будет превышать 24 часа
  • Максимум 4 раза для каждого лекарства (всего 8 максимум)
  • Входные данные всегда будут непустыми и содержат как минимум одно лекарство и один раз

Примеры:

Две дозы каждого с двухчасовым интервалом:

"P: 00:00, I: 02:00, P: 04:00, I: 06:00" -> "Next P: 08:00, I: 10:00"

Разовая доза парацетамола

"P: 22:00" -> "Next P: 02:00"

Максимальная доза парацетамола в течение 24 часов, разовая доза ибупрофена

"P: 04:05, P: 08:10, P: 12:15, I: 12:30, P: 16:25" -> "Next I: 16:30, P: 04:05"

Тестовые случаи:

"I: 06:00" -> "Next I: 10:00"
"P: 22:00" -> "Next P: 02:00"
"P: 22:00, P: 02:00, I: 06:00" -> "Next P: 06:00, I: 10:00"
"P: 00:00, I: 02:00, P: 04:00, I: 06:00" -> "Next P: 08:00, I: 10:00"
"P: 04:05, P: 08:10, P: 12:15, I: 12:30, P: 16:25" -> "Next I: 16:30, P: 04:05"
"I: 06:32, P: 08:15, I: 10:44, P: 13:03" -> "Next I: 14:44, P: 17:03"
"P: 07:30, I: 07:30, P: 11:30, I: 11:30, P: 15:30, I: 15:30, I: 19:30" -> "Next P: 19:30, I: 07:30"
"I: 07:30, P: 11:30, I: 11:30, P: 15:30, I: 15:30, P: 19:30, I: 19:30" -> "Next P: 23:30, I: 07:30"
"P: 07:30, I: 07:30, P: 11:30, I: 11:30, P: 15:30, I: 15:30, P: 19:30, I: 19:30" -> "Next P: 07:30, I: 07:30" OR "Next I: 07:30, P: 07:30"

Это код гольф, поэтому выигрывает кратчайший ответ в байтах.

ОБНОВИТЬ:

На выходе теперь могут быть аббревиатуры от парацетамола и ибупрофена; PиI

Erresen
источник
Было бы неплохо иметь некоторые рычаги влияния на формат ввода и вывода - мета-пост
Гурупад Мамадапур
Выход @GurupadMamadapur может быть, но извлечение времени и типа лекарства из ввода - половина проблемы
Erresen
Я бы порекомендовал вам разрешить людям сокращать количество парацетамола и ибупрофена в выходных данных, так как они добавляют ненужную длину в материалы
Cyoce
@Cyoce да, я согласен, пробую решение самостоятельно, и это на самом деле немного сложно - обновил правила, чтобы разрешить сокращенный вывод
Erresen
@ Линн согласилась и обновилась
Erresen

Ответы:

4

JavaScript (ES6), 367 362 354 358 байт

Гольф версия:

A=i=>i>9?""+i:"0"+i,B=(s,a=":")=>s.split(a),C=(a,b,c,d)=>[...[s,t]=B((b>3?c:d)||":"),a+` ${A(s=b>3?+s:(+s+4)%24)}:`+A(t=+t)],F=s=>{a=B(s,m=", ");for(b=c=d=e=f=p=q=0;f<a.length;g=="P:"?(b++,d=d?h:p=h):(c++,e=e?h:q=h))[g,h]=B(a[f++]," ");[i,j,k]=C("P",b,p,d),[n,o,l]=C("I",c,q,e),r=B(h)[0];return"Next "+c?b?n*60+(n<r)*1440+j<i*60+(i<r)*1440+o?l+m+k:k+m+l:l:k}

Ungolfed / прокомментировал:

// Returns a zero-padded string of the argument.
A=i=>i>9?""+i:"0"+i,

// Since we do a lot of splitting, alias it. Making the
// second argument optional (and defaulting to ':') saved
// 3 bytes
B=(s,a=":")=>s.split(a),

// Constructs a string for output, along with the time
// of the next dose, in the format [hour, minute, string].
// Arguments:               type
// a -> type (P/I)          String
// b -> amount of doses     Number
//      taken
// c -> first dose taken    String
// d -> last dose taken     String
//
// The first two values are split from the string, but
// before the array is returned, they are converted to
// integers (during the string construction).
C=(a,b,c,d)=>[...[s,t]=B((b>3?c:d)||":"),a+` ${A(s=b>3?+s:(+s+4)%24)}:`+A(t=+t)],

// Main function. Returns the time(s) for the next dose.
// Argument:                type
// s -> list of times of    String
//      and types of 
//      doses taken
F=s=>{
    a=B(s,m=", "); // Split the input by comma + space,
                   // and save that string, since we
                   // need it later when constructing
                   // the output string.
    // For loop has been restructured. Original:
    // for(b=c=f=0;f<a.length;g=="P:"?(b++,d=d?h:p=h):(c++,e=e?h:q=h))
    //     [g,h]=B(a[f++]," ");
    b = 0; // P-dose counter
    c = 0; // I-dose counter
    d = 0; // Last P-dose
    e = 0; // Last I-dose
    p = 0; // First P-dose
    q = 0; // First I-dose
    for (f = 0; f < a.length; f++) {
        [g, h] = B(a[f], " ");  // g contains the type,
                                // h contains the time
        if (g == "P:") {
            b++;                // increase the counter

            if (d == 0) {   // store h in p if this is
                p = h;      // the first dose of this
            }               // type
            d = h;
        } else {
            // See the above code block for comments
            c++;

            if (e == 0) {
                q = h;
            }
            e = h;
        }
    }
    // End of restructured for loop.

    // Construct the output strings, and get the times.
    // See comments at C function.
    [i, j, k] = C("P", b, p, d);
    [n, o, l] = C("I", c, q, e);

    // Get the amount of hours of the dose taken last.
    // We use this to get the correct order of the two
    // times.
    r = B(h)[0];

    // Return statement has been restructured. Original:
    // return "Next "+c?b?n*60+(n<r)*1440+j<i*60+(i<r)*1440+o?l+m+k:k+m+l:l:k
    //==================================================
    // Start creating the output string.
    output = "Next "
    // Use the following checks to figure out what needs
    // to be part of the output and in what order.
    if (c > 0) {
        if (b > 0) {
            // Compare the times of next doses
            // P_time = (i + (i < r) * 24) * 60
            // I'm using implicit conversion of
            // booleans to numbers. If the next
            // dose is past midnight, add 1 * 24
            // to the time, so it is compared
            // correctly.
            // Then add the minutes to the number.
            P_time = i*60+(i<r)*1440+o;
            I_time = n*60+(n<r)*1440+j;

            if (I_time < P_time) {
                output += l + m + k; // I first
            } else {
                output += k + m + l; // P first
            }
        } else {
            output += l; // Just I
        }
    } else {
        output += k; // Just P
    }

    // Finally, return the output
    return output;
}

Чтобы использовать его, вызовите F со строкой в ​​качестве аргумента следующим образом:

F("P: 04:00, I: 06:00")
Люк
источник
Это здорово, но у меня была пара проблем. Кажется, что это не сработает, если на входе присутствует только один тип таблеток, например F("P: 22:00")-> ReferenceError: q is not defined. Этот ввод будет выполняться, если на P & I уже ссылались ранее, но со старыми деталями для I.
Крис М
Благодарность! Я только что проверил это, и вы правы насчет ошибки ссылки. Я предполагаю, что переменная q не сбрасывается, и я не уделял достаточного внимания во время тестирования. Спасибо, что сообщили мне, я исправлю это позже.
Люк
Оказалось, что это легко исправить, но это обошлось мне в 4 байта.
Люк
1

Python 3 - 437 байт

a=input();i=p=l=-1;j=q=0
for x in a.split(", ")[::-1]:
    for y, z in [x.split(": ")]:
        s=lambda q,r,t:[t,sum([a*b for a,b in zip([60,1],map(int,q.split(':')))])][r%4<2]+[0,240][r<2]
        if y=="I":j+=1;i=s(z,j,i)
        else:q+=1;p=s(z,q,p)
        l=[l,p+i-239][j+q<2]
r=lambda d,e:("","%s: %02d:%02d, "%(d,(e/60)%24,e%60))[e>-1];p+=[1440,0][p>=l];i+=[1440,0][i>=l];print("Next "+[r("I",i)+r("P",p),r("P",p)+r("I",i)][p<i][:-2])

Объяснение:

a=input();i=p=l=-1;j=q=0
for x in a.split(", ")[::-1]: #Read in reverse order, a="P: 01:00"
    for y, z in [x.split(": ")]:#Y="P", Z="00:00"
        s=
        lambda q,r,t:[t,sum([a*b for a,b in zip([60,1],map(int,q.split(':')))])]#Convert "01:01" to 61
        [r%4<2]#In case it's the first or fourth string calculate a new value, otherwise: return the original value
        +[0,240][r<2]#In case it's the last string: add 4 hours. Otherwise, leave it.
        if y=="I":j+=1;i=s(z,j,i)#Calculate for i
        else:q+=1;p=s(z,q,p)#Calculate for p
        l=[l,p+i-239][j+q<2]#Sets the last record. Since we read in reverse order, this should be the first one. We've added 4 hours though so remove those again
r=lambda d,e:("","%s: %02d:%02d, "%(d,(e/60)%24,e%60))[e>-1];#Print function, don't print anything when we have no value
p+=[1440,0][p>=l];i+=[1440,0][i>=l];    #Add a day if record is before the last record so we can correctly calculate the order
print("Next "+[r("I",i)+r("P",p),r("P",p)+r("I",i)][p<i][:-2])#print it and remove the last ","
Карра
источник
1

PHP, 228 241 239 227 226 байт

требует PHP 7

Next<?foreach(explode(", ",$argv[1])as$d){[$m,$h,$i]=explode(":",$d);$x[$m][++$$m]=24+$h+$i/60;}foreach($x as$m=>$d)$r[$m]=$d[$$m-3]?:$d[$$m]-20;sort($r);foreach($r as$m=>$t)$o[]=" $m: ".date("i:s",$t%24*60);echo join(",",$o);

сломать

Next<?                              // print "Next"
foreach(explode(", ",$argv[1])as$d) // loop through string split by comma+space
{
    [$m,$h,$i]=explode(":",$d);         // separate drug, hours and minutes
    $x[$m][++$$m]=24+$h+$i/60;          // append time to array, track count in ${$m}
}                                       // (i.e. $P for drug "P" etc.)
foreach($x as$m=>$d)                // loop through drugs
    $r[$m]=                             // add time to result
        $d[$$m-3]                           // if more than 3 medications, use $$m-3
            ??$d[$$m]-20                    // else use last medication - 20 hours
    ;
sort($r);                           // sort results by time
foreach($r as$m=>$t)$o[]=" $m: "    // prepare for output: drug name and formatted time:
    .date("i:s",$t%24*60)           // use hrs as mins and mins as secs to avoid TZ problems
;
echo join(",",$o);                  // print
Titus
источник
0

JavaScript (ES6), 246 байт

s=>s.split`, `.map(s=>(m[s[0]].unshift(t=s.replace(/\d+/,h=>(h=(1+h)%24)>9?h:`0`+h),s),l=l||t.slice(1)),l=0,m={I:[],P:[]})&&`Next `+[].concat(m.I[7]||m.I[0]||[],m.P[7]||m.P[0]||[]).sort((i,p)=>((i=i.slice(1))<l)-((p=p.slice(1))<l)||i>p).join`, `

Объяснение:

Циклическое над каждой дозы, как Iи Pдозы разделены на два массива. 4 часа также добавляются к каждой дозе, и эти времена также сохраняются. Массивы заполняются в обратном порядке, чтобы упростить обнаружение 8 записей. Время 4 часа после первой дозы также сохраняется для использования во время сортировки. В этот момент каждый массив может находиться в одном из трех состояний:

  • 8 записей, и в этом случае последняя запись является первой дозой, а следующая доза должна быть через 24 часа после этой дозы (то есть в то же время завтра)
  • 2, 4 или 6 записей, в этом случае первая запись составляет 4 часа после последней дозы, и, следовательно, время следующей дозы
  • 0 записей, в этом случае мы объединяем [], что выравнивается и, следовательно, исключается из результата

Извлекая время следующей дозы из двух массивов, осталось отсортировать их по порядку. Это делается путем сравнения их со временем через 4 часа после первой дозы. Если один из двух периодов до этого времени, это должно относиться к завтрашнему дню, и эта доза приходит последней. В противном случае времена просто сравниваются напрямую. (Скорее неудобно, лекарство раньше времени, поэтому я должен раздеть его, чтобы сравнить.)

Нил
источник