Pomóż mi zarządzać czasem

15

Niedawno powiedziano mi, żebym przeczytał cały podręcznik fizyki do nowego roku (prawdziwa historia, niestety). Potrzebuję twojej pomocy w ustaleniu, które rozdziały powinienem czytać każdego dnia. Właśnie tam wchodzisz.

Wejście

  • Dwie daty, w dowolnym formacie. Druga data zawsze będzie późniejsza niż pierwsza.
  • Lista numerów rozdziałów. Ta lista oddzielona przecinkami może zawierać pojedyncze rozdziały ( 12) lub zakresy włącznie ( 1-3). Dawny. 1-3,5,6,10-13.
  • Lista dni tygodnia (reprezentowana przez dwie pierwsze litery nazwy Monday -> Mo:) do wykluczenia z harmonogramu. Dawny. Mo,Tu,Fr.

Wynik

Wyjściem będzie rozdzielona nowym wierszem lista dat i numerów rozdziałów (patrz format poniżej). Rozdziały powinny być równomiernie rozmieszczone we wszystkich dniach w zakresie, z wyłączeniem przewidzianych dni tygodnia. Jeśli rozdziały nie rozkładają się równomiernie, należy mieć dni z niższą liczbą rozdziałów na koniec okresu. Daty wyjściowe mogą mieć inny format niż wejściowy. Dni bez rozdziałów można pominąć lub po prostu nie zawierać żadnych rozdziałów.

Przykład:

Wejście: 9/17/2015 9/27/2015 1-15 Tu

Wynik:

9/17/2015: 1 2
9/18/2015: 3 4
9/19/2015: 5 6
9/20/2015: 7 8
9/21/2015: 9 10
9/23/2015: 11
9/24/2015: 12
9/25/2015: 13
9/26/2015: 14
9/27/2015: 15
GamrCorps
źródło
Dane wejściowe w tym przykładzie powinny brzmieć „9/17/2015 9/27/2015 1-15 Tu”, ponieważ 9/22 to wtorek.
DavidC,
@DavidCarraher masz rację, kiedy wprowadziłem ten przykładowy materiał, z jakiegoś powodu myślałem o listopadzie.
GamrCorps,
7
Gdybym to był ostatni dzień, miałby wszystkie rozdziały :)
MickyT,
@MickyT właśnie moja inspiracja do tego wyzwania.
GamrCorps,
Wkrótce odkryjesz, jak niesamowita jest fizyka. Właściwie masz szczęście.
Fabrizio Calderan

Odpowiedzi:

2

PowerShell v4, 367 357 323 313 308 307 305 277 bajtów

param($a,$b,$c,$d)$e=@();$c=-split('('+($c-replace'-','..'-replace',','),(')+')'|iex|%{$_-join' '});while($a-le$b){if(-join"$($a.DayOfWeek)"[0,1]-notin$d){$e+=$a;$z++}$a=$a.AddDays(1)}$g=,0*$z;$c|%{$g[$c.IndexOf($_)%$z]++};1..$z|%{"$($e[$_-1]): "+$c[$i..($i+=$g[$_-1]-1)];$i++}

Edytuj - golfowano 28 bajtów przy użyciu jawnego formatowania wejściowego.

Wyjaśniono:

param($a,$b,$c,$d)    # Parameters, takes our four inputs
$e=@()                # This is our array of valid output dates

$c=-split('('+($c-replace'-','..'-replace',','),(')+')'|iex|%{$_-join' '})
# Ridiculously complex way to turn the input chapters into an int array
# The first part changes "1,5-9,12" into a "(1),(5..9),(12)" format that
# PowerShell understands, then executes that with iex, which creates an 
# array of arrays. Then iterate through each inner array and joins them all
# together with spaces, then finally splits on spaces to create a 1D array

while($a-le$b){       # Until we reach the end day
  if(-join"$($a.DayOfWeek)"[0,1]-notin$d){
    # Not an excluded day of the week
    $e+=$a            # Add it to our list of days
    $z++              # Increment our count of total days
  }
  $a=$a.AddDays(1)    # Move to the next day in the range
}

$g=,0*$z              # Populate a new array with zeroes, same length as $e

$c|%{$g[$c.IndexOf($_)%$z]++}
# This populates $g for how many chapters we need each day

1..$z|%{"$($e[$_-1]): "+$c[$i..($i+=$g[$_-1]-1)];$i++}
# Goes through the days in $e, prints them, and slices $c based on $g

Stosowanie

Oczekuje, że daty będą w DateTimeformacie .NET . Oczekuje, że „pominięte” dni będą w tablicy (odpowiednik listy programu PowerShell).

PS C:\Tools\Scripts\golfing> .\help-me-manage-my-time.ps1 (Get-Date '9/17/2015') (Get-Date '9/27/2015') '5,1-3,6,10-13,20-27' @('Su','Tu')
09/17/2015 00:00:00: 5 1 2
09/18/2015 00:00:00: 3 6
09/19/2015 00:00:00: 10 11
09/21/2015 00:00:00: 12 13
09/23/2015 00:00:00: 20 21
09/24/2015 00:00:00: 22 23
09/25/2015 00:00:00: 24 25
09/26/2015 00:00:00: 26 27
AdmBorkBork
źródło
3
Jest tak wiele znaków dolara ... To musi być drogie! : D
kirbyfan64sos,
@ kirbyfan64sos Tylko 12% znaków tutaj to $... To naprawdę całkiem średnia dla golfa PowerShell, który wydaje się być około 10% do 15% (na podstawie moich nieformalnych obliczeń odpowiedzi, które opublikowałem).
AdmBorkBork,
Do ciebie jeszcze raz :-)
Willem
Hmm 308 również dla mojego ...
Willem,
Dobra robota na 305! 300 teraz :-)
Willem
3

JavaScript (ES6), 317 310 291 bajtów

(a,b,c,d)=>{u=0;c.split`,`.map(m=>{p=m[s]`-`;for(q=n=p[0];n<=(p[1]||q);r=++u)c+=","+n++},c="");c=c.split`,`;x=d.map(p=>"SuMoTuWeThFrSa".search(p)/2);for(g=[];a<b;a.setTime(+a+864e5))x.indexOf(a.getDay())<0&&(t=y=g.push(a+" "));return g.map(w=>w+c.slice(u-r+1,u-(r-=r/y--+.99|0)+1)).join`
`}

Stosowanie

f(new Date("2015-09-17"),new Date("2015-09-27"),"5,1-4,6,10-13,20-27",["Su","Tu"])
=> "Thu Sep 17 2015 10:00:00 GMT+1000 (AUS Eastern Standard Time) 5,1,2
Fri Sep 18 2015 10:00:00 GMT+1000 (AUS Eastern Standard Time) 3,4,6
Sat Sep 19 2015 10:00:00 GMT+1000 (AUS Eastern Standard Time) 10,11
Mon Sep 21 2015 10:00:00 GMT+1000 (AUS Eastern Standard Time) 12,13
Wed Sep 23 2015 10:00:00 GMT+1000 (AUS Eastern Standard Time) 20,21
Thu Sep 24 2015 10:00:00 GMT+1000 (AUS Eastern Standard Time) 22,23
Fri Sep 25 2015 10:00:00 GMT+1000 (AUS Eastern Standard Time) 24,25
Sat Sep 26 2015 10:00:00 GMT+1000 (AUS Eastern Standard Time) 26,27"

Wyjaśnienie

(a,b,c,d)=>{

  u=0;                                                 // u = total chapters
  c.split`,`.map(m=>{                                  // c = array of each chapter
    p=m[s]`-`;
    for(q=n=p[0];n<=(p[1]||q);r=++u)                   // get each chapter from ranges
      c+=","+n++
  },c="");
  c=c.split`,`;

  x=d.map(p=>"SuMoTuWeThFrSa".search(p)/2);            // x = days to skip
  for(g=[];a<b;a.setTime(+a+864e5))                    // for each day between a and b
    x.indexOf(a.getDay())<0&&                          // if this day is not skipped
      (t=y=g.push(a+" "));                             // add it to the list of days
                                                       // t = total days
                                                       // y = days remaining

  return g.map(w=>w+
    c.slice(u-r+1,u-(r-=r/y--+.99|0)+1)                // add the chapters of the day
  ).join`
`
}
użytkownik 81655
źródło
2

Python 2 - 338 317 308 304 300

Idziemy, żeby piłka się toczyła ...

def f(a,b,c,d):
 from pandas import*;import numpy as n
 s=str.split;e=n.array([])
 for g in s(c,','):h=s(g,'-');e=n.append(e,range(int(h[0]),int(h[-1])+1))
 k=[t for t in date_range(a,b) if s('Mo Tu We Th Fr Sa Su')[t.weekday()]not in d];j=len(k);e=array_split(e,j)
 for u in range(j):print k[u],e[u]

Przykładowe dane wejściowe:

f('9/17/2015','9/27/2015','5,1-3,6,10-13,20-27',['Su','Tu'])

Przykładowe dane wyjściowe:

2015-09-17 00:00:00 [ 5.  1.  2.]
2015-09-18 00:00:00 [ 3.  6.]
2015-09-19 00:00:00 [ 10.  11.]
2015-09-21 00:00:00 [ 12.  13.]
2015-09-23 00:00:00 [ 20.  21.]
2015-09-24 00:00:00 [ 22.  23.]
2015-09-25 00:00:00 [ 24.  25.]
2015-09-26 00:00:00 [ 26.  27.]
Willem
źródło