Data wystąpienia

9

Biorąc pod uwagę trzy nieujemne liczby całkowite y, moraz d(z których co najmniej jeden musi być dodatnia) i poprawną datę z pozytywnym roku (w dowolnym odpowiednim formacie, który obejmuje rok, miesiąc i dzień, a nie informacje dodatkowe), wyjście z data, to jest ylata, mmiesiące i ddni po pierwotnej dacie.

Kalendarza gregoriańskiego należy używać dla wszystkich dat (nawet dat poprzedzających przyjęcie kalendarza gregoriańskiego).

Metoda obliczania następnej daty jest następująca:

  1. Dodaj ydo roku
  2. Dodaj mdo miesiąca
  3. Normalizuj datę, stosując najazdy (np. 2018-13-01-> 2019-01-01)
  4. Jeśli dzień minął ostatni dzień miesiąca, zmień go na ostatni dzień miesiąca (np. 2018-02-30-> 2018-02-28)
  5. Dodaj ddo dnia
  6. Normalizuj datę, stosując najazdy (np. 2019-01-32-> 2019-02-01)

Lata przestępne (lata podzielne przez 4, ale niepodzielne przez 100, chyba że podzielne przez 400) muszą być odpowiednio obsługiwane. Wszystkie wejścia i wyjścia będą w reprezentatywnym zakresie liczb całkowitych twojego języka.

Przypadki testowe

Przypadki testowe są dostarczane w formacie input => output, w którym inputznajduje się obiekt JSON.

{"date":"2018-01-01","add":{"d":1}} => 2018-01-02
{"date":"2018-01-01","add":{"M":1}} => 2018-02-01
{"date":"2018-01-01","add":{"Y":1}} => 2019-01-01
{"date":"2018-01-30","add":{"M":1}} => 2018-02-28
{"date":"2018-01-30","add":{"M":2}} => 2018-03-30
{"date":"2000-02-29","add":{"Y":1}} => 2001-02-28
{"date":"2000-02-29","add":{"Y":4}} => 2004-02-29
{"date":"2000-01-30","add":{"d":2}} => 2000-02-01
{"date":"2018-01-01","add":{"Y":2,"M":3,"d":4}} => 2020-04-05
{"date":"2018-01-01","add":{"Y":5,"M":15,"d":40}} => 2024-05-11

Możesz użyć tego JSFiddle do testowania.

To jest , więc wygrywa najkrótsze rozwiązanie (w każdym języku).

Mego
źródło
Wpis w piaskownicy (usunięty)
Mego,
2
@LuisfelipeDejesusMunoz Format wejściowy nie jest ważny, podobnie jak w przypadku PPCG.
Mego
Czy istnieją jakiekolwiek ograniczenia w górnych granicach y, ma d(np mogłyby dbyć 2147483000?)
ErikF
@ErikFAll inputs and outputs will be within the representable integer range of your language.
Mego
1
Co z formatami wyjściowymi? Czy możemy wyprowadzić obiekt daty? Czy możemy wziąć obiekt daty?
Asone Tuhid

Odpowiedzi:

3

C (gcc) , 291 bajtów

Ten był całkiem fajny, gdy zwracał te same wartości, co wbudowany JS.

z,m=0xEEFBB3;int*y;g(){z=28+(m>>y[1]*2&3)+!(y[1]-1)*(!(*y%4)&&(*y%100)||!(*y%400));}h(a){z=(a>g())?g():a;}j(){*y+=y[1]/12;y[1]%=12;y[2]=h(y[2]);}f(int*a){y=a+6;for(z=0;z<3;z++)y[z]=a[z];y[1]--;j();*y+=a[3];y[1]+=a[4];j();y[2]+=a[5];for(;y[2]>h(y[2]);(y[1]=++y[1]%12)||++*y)y[2]-=g();y[1]++;}

Wypróbuj online!

Bez golfa:

// De No Oc Se Au Jl Jn Ma Ap Mr Fe Ja
// 31 30 31 30 31 31 30 31 30 31 28 31 = Month length
// 11 10 11 10 11 11 10 11 10 11 00 11 = Offset (2-bit representation)
//   E     E     F     B     B     3   = Hex representation

int m=0xEEFBB3; // Month lengths-28 in reverse order, stored as 2 bits/month
int *y; // Pointer to the output date, shared as a global between calls

// Regenerate month length and add leap day
int days_month(void) { 
  return 28+(m>>y[1]*2&3)+!(y[1]-1)*(!(*y%4)&&(*y%100)||!(*y%400));
}

int calendar_day(int day) { return day>days_month()?days_month():day; }

void truncate_date(void) {
  *y+=y[1]/12; y[1]%=12;
  y[2]=calendar_day(y[2]);
}

void f(int *a) {
  int z;
  y=a+6;
  for(z=0;z<3;z++)y[z]=a[z];y[1]--; // Convert month to 0-based
  truncate_date();
  *y+=a[3]; y[1]+=a[4]; truncate_date();
  y[2]+=a[5];
  for(;y[2]>calendar_day(y[2]);(y[1]=++y[1]%12)||++*y)
    y[2]-=days_month();
  y[1]++; // Return month to 1-based
}

Wypróbuj online!

ErikF
źródło
249 bajtów
pułap pułapu
1

perl -MDate :: Calc =: all -E, 28 bajtów

$,=$";say Add_Delta_YMD@ARGV

Wymaga to 6 argumentów: roku wprowadzenia, miesiąca i daty (jako osobnych argumentów) oraz liczby lat, miesięcy i dni do dodania.


źródło
2
Nie dotyczy to dziwnej „reguły 4” zadania, więc zawodzi niektóre przypadki testowe - np. perl -MDate::Calc=:all -E '$,=$";say Add_Delta_YMD@ARGV' -- 2000 2 29 1 0 0Zwraca 2001 3 1zamiast tego, 2001 2 28jak oczekuje OP (przypadek testowy 6).
Sundar - Przywróć Monikę
1

R , 88 bajtów

function(Y,M,D,y,m,d,o=M+m){while(is.na(x<-ISOdate(Y+y+o%/%12,o%%12,D)))D=D-1;x+864e2*d}

Wypróbuj online!

Funkcja, która pobiera 3 argumenty ( Y,M,D) dla daty i inne 3 argumenty ( y,m,d) dla wartości, które mają zostać dodane.

Wyjście jest wyposażony poprzedzany 12:00:00 GMTktóry jest domyślnym formatem dla ISOdate„s

digEmAll
źródło
1

Perl 6 ,  60 50 45  44 bajtów

{Date.new($^a).later(:$:year).later(:$:month).later(:$:day)}

Sprawdź to (60)
Wejście jest( "2000-02-29", year => 1, month => 0, day => 0 )


{$^a.later(:$:year).later(:$:month).later(:$:day)}

Sprawdź to (50)
Wejście jest( Date.new("2000-02-29"), year => 1, month => 0, day => 0 )


{$/=$^a;$/.=later(|$_) for |[R,] $^b.sort;$/}

Sprawdź to (45) Dane
wejściowe to ( Date.new("2000-02-29"), %( year => 1 ) )
(Nie trzeba dołączać kluczy o wartości 0)


{$/=$^a;$/.=later(|$_) for |[R,] %_.sort;$/}

Sprawdź to (44)
Wejście jest( Date.new("2000-02-29"), year => 1 )

Rozszerzony:

{  # bare block lambda

  $/ = $^a; # store only positional param into a modifiable scalar
            # (params are readonly by default)


  # do a loop over the data to add

  $/ .= later(    # add using Date.later()
    |$_           # turn current iterated Pair into a named parameter
  )

    for

      |           # flatten so that `for` will iterate

        [R,]      # shorter than `reverse` (year=>1, month=>0, day=>0)

          %_.sort # sort the named arguments (day=>0, month=>0, year=>1)
  ;

  # return new Date
  $/
}
Brad Gilbert b2gills
źródło
Możesz usunąć to miejsce przedfor
Jo King
1

Java 8, 51 bajtów

(s,y,m,d)->s.plusYears(y).plusMonths(m).plusDays(d)

Input ( s) i output są jednocześnie java.time.LocalDate.

Wypróbuj online.

Wyjaśnienie:

(s,y,m,d)->        // Method with LocalDate and 3 int parameters and LocalDate return-type
  s.plusYears(y)   //  Add the years to the input start-Date
   .plusMonths(m)  //  Add the months as well
   .plusDays(d)    //  And add the days as well
Kevin Cruijssen
źródło
1

R , 65 bajtów

function(x,y){require(lubridate)
x%m+%period(y,c("ye","mo","d"))}

Używa lubridatepaczki. %m+%Operator infix jest cukier do add_with_rollbackfunkcji, która implementuje co zasadniczo kwestia prosi.

TIO nie ma, lubridatewięc możesz wypróbować tutaj zamiast tego z dodatkiem f <-do powyższej funkcji i przypadków testowych:

f(as.Date("2018-01-01"),c(0,0,1))
f(as.Date("2018-01-01"),c(0,1,0))
f(as.Date("2018-01-01"),c(1,0,0))
f(as.Date("2018-01-30"),c(0,1,0))
f(as.Date("2018-01-30"),c(0,2,0))
f(as.Date("2000-02-29"),c(1,0,0))
f(as.Date("2000-02-29"),c(4,0,0))
f(as.Date("2000-01-30"),c(0,0,2))
f(as.Date("2018-01-01"),c(2,3,4))
f(as.Date("2018-01-01"),c(5,15,40))
ngm
źródło
Możesz zapisać dwa bajty za pomocą: function(x,y)x%m+%period(y,c("ye","mo","d")) require(lubridate)(wymagają poza funkcją)
JayCe
0

Bash , 150 149 bajtów

a=$2+$5-1+b
y=$1+$4+a/12
m=1+a%12
d=date
$d -d@$[$($d +%s+$6*86400 -d$[y]-$[m]-$($d +$3%n%d -d@$[`b=1;$d +%s-86400 -d$[y]-$[m]-1`]|sort -n|head -1))]

Wypróbuj online!

Pobiera dane wejściowe za pomocą argumentów wiersza poleceń w kolejności: stary rok, stary miesiąc, stary dzień. zmiana roku, zmiana miesiąca, zmiana dnia. Zwraca ciąg podobny Wed Feb 28 00:00:00 UTC 2018do standardowego.

pustkowie
źródło
0

PHP , 203 bajty

<?list(,$a,$y,$m,$d)=$argv;$b=new DateTime($a);$j=$b->format('j');$b->modify("+$y year +$m month");$j==$b->format('j')?:$b->modify('last day of last month');$b->modify("+$d day");echo$b->format('Y-m-d');

Aby uruchomić:

php -n <filename> <date> <y> <m> <d>

Przykład:

php -n date_occurrences.php 2018-01-01 5 15 40

Lub wypróbuj online!

Testy: Wypróbuj online!

Noc 2
źródło
0

T-SQL, 53 bajty

SELECT DATEADD(D,d,DATEADD(M,m,DATEADD(Y,y,a)))FROM t

Nie jestem pewien, czy to ma znaczenie, ale stosuję korektę roku, a następnie korektę miesiąca, a następnie dzień. Wszystkie wartości testowe sprawdź.

Zgodnie z naszymi standardami IO , wejście pochodzi z wcześniej istniejących tabeli T z datą pola A i całkowitą pola y , m i d .

Co ciekawe, to nie wielkość liter ma znaczenie między kodami typu daty ( D , M i Y ) a moimi wartościami wejściowymi ( d , m i y ), lecz po prostu kolejność parametrów w DATEADDfunkcji SQL .

BradC
źródło
1
Czy to zdaje test 6? Ponieważ nie implementuje Reguły 4, myślę, że dałby 2001 3 1zamiast 2001 2 28wkładu 6.
Sundar - Przywróć Monikę
@sundar Wygląda na to, że masz rację; Myślałem, że zdałem wszystkie testy. Zobaczę, czy można to naprawić ...
BradC