休日の頭痛


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
  • (入力文字列で)服用している薬が1つだけの場合、出力文字列にはその薬だけが含まれている必要があります。 Next P: 02:00

ルール:

  • パラセタモール「P」とイブプロフェン「I」の2種類の薬しかありません。
  • パラセタモールは4時間ごとに1回、24時間以内に最大4回服用できます。
  • イブプロフェンは、4時間に1回、24時間以内に最大4回服用することもできます。
  • パラセタモールとイブプロフェンは一緒に摂取することも、別々に摂取することもできます。一方は他方の投与量にはカウントされません。
  • 入力文字列の時間は常に連続しますが、深夜(23:00-> 03:00)を超える場合があります。
  • 入力文字列の時間は24時間を超えない
  • 各薬剤につき最大4回(合計で最大8回)
  • 入力は常に空ではなく、少なくとも1つの薬物と1つの時間を含みます

例:

2時間間隔でそれぞれ2回の投与:

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

これはコードゴルフであるため、intバイトの最短回答が優先されます。

更新:

出力は、パラセタモールとイブプロフェンの略語になります。PそしてI


-私は、入力と出力のフォーマットにいくつかの影響力を持っている素敵なことだメタポスト
Gurupad Mamadapur

@GurupadMamadapur多分出力が、入力からの時間と薬の種類を抽出することは半分の挑戦である
Erresen

提出物に不要な長さを追加するため、出力でパラセタモールとイブプロフェンを省略できるようにすることをお勧めします
Cyoce

@Cyoceええ、私は解決策を自分でしようとし、同意し、それが実際には少しトリッキーだ-略し出力できるようにするルールを更新
Erresen

@Lynnが同意し、更新しました
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}

非ゴルフ/コメント:

// 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")

これは素晴らしいことですが、いくつかの問題がありました。入力に錠剤の種類が1つしかない場合、たとえばF("P: 22:00")->の場合、失敗するようReferenceError: q is not definedです。P&Iは、しかしI.の古い詳細は、以前に参照されている場合は、この入力が実行されます
クリスM

ありがとう!私はそれをテストしましたが、参照エラーについては正しいです。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

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`, `

説明:

各投与量をループして、投与量IP投与量は2つの配列に分けられます。4時間も各投与に追加され、それらの時間も節約されます。8つのエントリを検出しやすくするために、配列は逆に配置されます。最初の投与から4時間後の時間も、ソート中に使用するために保存されます。この時点で、各配列は次の3つの状態のいずれかになります。

  • 8エントリ。この場合、最後のエントリは最初の投与であり、次の投与はこの投与の24時間後でなければなりません(つまり明日同じ時間)
  • 2、4、または6エントリ。この場合、最初のエントリは最後の投与の4時間後、したがって次の投与の時間です。
  • 0エントリ。この場合、concatentate []がフラット化され、結果から除外されます。

2つのアレイから次の投与時間を抽出した後、それらを順番にソートする必要があります。これは、最初の投与の4時間後と比較して行われます。2回のうちの1回がこの時間より前の場合、これは明日を指している必要があり、その用量は最後になります。それ以外の場合、時間は単純に直接比較されます。(むしろ不便なことに、薬は時間の前にあるので、適切に比較するためにそれを取り除く必要があります。)

弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.