Ocorrências de Data

9

Dada três inteiros não negativos y, me d(dos quais pelo menos um deve ser positivo) e uma data válida com um ano positivo (em qualquer formato razoável, que inclui o ano, mês e dia, e nenhuma informação adicional), a saída do data que é yanos, mmeses e ddias após a data original.

O calendário gregoriano deve ser usado para todas as datas (mesmo datas anteriores à adoção do calendário gregoriano).

O método para calcular a próxima data é o seguinte:

  1. Adicionar yao ano
  2. Adicionar mao mês
  3. Normalize a data aplicando rollovers (por exemplo, 2018-13-01-> 2019-01-01)
  4. Se o dia tiver passado o último dia do mês, altere-o para o último dia do mês (por exemplo, 2018-02-30-> 2018-02-28)
  5. Adicionar dao dia
  6. Normalize a data aplicando rollovers (por exemplo, 2019-01-32-> 2019-02-01)

Os anos bissextos (anos divisíveis por 4, mas não divisíveis por 100, a menos que também divisíveis por 400) devem ser tratados adequadamente. Todas as entradas e saídas estarão dentro do intervalo inteiro representável do seu idioma.

Casos de teste

Os casos de teste são fornecidos no formato input => output, onde inputé um objeto 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

Você pode usar este JSFiddle para teste.

Isso é , então a solução mais curta (em cada idioma) vence.

Mego
fonte
Postagem na caixa de areia (excluída)
Mego
2
@LuisfelipeDejesusMunoz O formato de entrada não é importante, como é a norma aqui no PPCG.
Mego
Existe alguma restrição para os limites superiores de y, me d(por exemplo, poderia dser 2147483000?)
ErikF
@ErikFAll inputs and outputs will be within the representable integer range of your language.
Mego
11
E quanto aos formatos de saída? Podemos produzir um objeto de data? Podemos pegar um objeto de data?
Asone Tuhid

Respostas:

3

C (gcc) , 291 bytes

Foi muito divertido obter os mesmos valores que o JS embutido.

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

Experimente online!

Sem golfe:

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

Experimente online!

ErikF
fonte
249 bytes
ceilingcat 30/07/2018
1

perl -MDate :: Calc =: todos -E, 28 bytes

$,=$";say Add_Delta_YMD@ARGV

São necessários 6 argumentos: o ano de entrada, o mês e a data (como argumentos separados) e o número de anos, meses e dias a serem adicionados.


fonte
2
Isso não lida com a "regra 4" peculiar da tarefa, portanto falha em alguns casos de teste - por exemplo, perl -MDate::Calc=:all -E '$,=$";say Add_Delta_YMD@ARGV' -- 2000 2 29 1 0 0retorna em 2001 3 1vez de 2001 2 28como o OP espera (caso de teste 6).
sundar - Restabelece Monica
1

R , 88 bytes

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}

Experimente online!

Uma função que usa 3 argumentos ( Y,M,D) para a data e outros 3 argumentos ( y,m,d) para os valores a serem adicionados.

A saída vem com anexo, 12:00:00 GMTque é o formato padrão para ISOdateos

digEmAll
fonte
1

Perl 6 ,  60 50 45  44 bytes

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

Teste (60) A
entrada é( "2000-02-29", year => 1, month => 0, day => 0 )


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

Teste (50) A
entrada é( Date.new("2000-02-29"), year => 1, month => 0, day => 0 )


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

Teste (45) A
entrada é ( Date.new("2000-02-29"), %( year => 1 ) )
(não é necessário incluir chaves com o valor 0)


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

Teste (44) A
entrada é( Date.new("2000-02-29"), year => 1 )

Expandido:

{  # 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
fonte
Você pode remover o espaço antes dafor
Jo King
1

Java 8, 51 bytes

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

Input ( s) e output são ambos java.time.LocalDate.

Experimente online.

Explicação:

(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
fonte
1

R , 65 bytes

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

Usa o lubridatepacote. O %m+%operador infix é o açúcar para a add_with_rollbackfunção que implementa essencialmente o que a pergunta pede.

O TIO não possui, lubridateportanto, você pode tentar aqui em vez disso, com f <-anexado à função acima junto com os casos de teste:

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
fonte
Você pode salvar salvar dois bytes com: function(x,y)x%m+%period(y,c("ye","mo","d")) require(lubridate)(exigem outisde de função)
Jayce
0

Bash , 150 149 bytes

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

Experimente online!

Recebe entrada por meio de argumentos de linha de comando em ordem: ano antigo, mês antigo, dia antigo. mudança de ano, mudança de mês, mudança de dia. Produz uma string como Wed Feb 28 00:00:00 UTC 2018stdout.

wastl
fonte
0

PHP , 203 bytes

<?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');

Para executá-lo:

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

Exemplo:

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

Ou Experimente online!

Testes: Experimente online!

Night2
fonte
0

T-SQL, 53 bytes

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

Não sei se isso importa, mas estou aplicando o ajuste de Ano, seguido pelo mês, seguido pelo dia. Todos os valores de teste são verificados.

De acordo com nossos padrões de IO , a entrada é obtida de uma tabela preexistente t com o campo de data a e o número inteiro y , m , e d .

Observe de maneira interessante que não é a capitalização que importa entre os códigos de tipo de data ( D , M e Y ) e meus valores de entrada ( d , m e y ) é simplesmente a ordem dos parâmetros na DATEADDfunção SQL .

BradC
fonte
11
Isso passa no caso de teste 6? Uma vez que não implementar Regra 4, acho que daria 2001 3 1em vez de 2001 2 28para a entrada de 6.
sundar - Reintegrar Monica
Parece que você está correto; Eu pensei que tinha passado em todos os casos de teste. Vou ver se ele pode ser corrigido ...
BradC