Como gerar uma data de trimestre do ano a partir da linha de comando?

10

Estou interessado em apresentar uma representação do atual trimestre do ano, bem como do trimestre do mês anterior.

Se hoje é 1º de janeiro de 2012, eu gostaria de obter

2012q1

e

2011q4

como as respectivas saídas.

smokris
fonte

Respostas:

10

Uma solução (meio feia), usando a avaliação aritmética do BASH e o datecomando GNU :

echo $(date +%Y)q$(( ($(date +%-m)-1)/3+1 ))
echo $(date -d "-1 month" +%Y)q$(( ($(date -d "-1 month" +%-m)-1)/3+1 ))

Observe que a opção %-mevita o datepreenchimento de 0, portanto isso ainda funcionará em agosto e setembro.

smokris
fonte
2
Pegue $ (...) em vez dos backticks obsoletos. Eles podem ser facilmente aninhados.
usuário desconhecido
11
apenas fique curioso: por que outras respostas usam 4 como divisor?
Liuyan刘研
2
@LiuYan 刘 研: Não tenho certeza. Há três meses em um quarto, então acho que 3 é o divisor correto. (Apesar de que me confuso no início, também.)
smokris
Em setembro, %m09que bash tenta interpretar como octal devido ao 0 inicial, então isso gera um erro que diz 09: value too great for base (error token is "09"). Isso pode ser corrigido desativando o preenchimento de 0, alterando %mpara %-m.
Matthew
5

Use meus dateutils :

dconv 2012-01-01 -f '%Y%Q'
=>
  2012Q1

Os sinalizadores %qe %Qsão específicos para dateutils e retornam o trimestre como número ou no formulário Q<NUMBER>.

hroptatyr
fonte
dconv now -f%Y%Q | tr Q qse você realmente precisa que o Q seja menor. (PS: estamos embalagem isso em Fedora com datecomo prefixo em vez de d, por isso " dateconv".)
mattdm
5

Todas as soluções divididas por quatro falham, por exemplo, novembro:

% echo $(( 11/4+1 ))
3

A matemática correta seria:

$(( (m-1)/3 +1 ))

E, como tal, o trimestre do mês atual e anterior seria:

echo curr ${y}q$(((m-1)/3+1))
if [ $m = 1 ]; then
  echo prev $((y-1))q4
else
  echo prev ${y}q$(((m-2)/3+1))
fi

São apenas doze valores para verificar, realmente ...

% for m in {1..12}; do echo $m Q$(((m-1)/3+1)); done
1 Q1
2 Q1
3 Q1
4 Q2
5 Q2
6 Q2
7 Q3
8 Q3
9 Q3
10 Q4
11 Q4
12 Q4
bif
fonte
11
+1 em "Todas as soluções divididas por falhas". Você está certo! Um último trimestre para 3 meses para que as respostas devem ser dividir por 3.
Stephen Quan
4

Provavelmente, não há solução direta.

Você pode usar awkpara evitar tantos tiques de volta.

date +"%Y %m" | awk '{q=int($2/4)+1; printf("%sq%s\n", $1, q);}'
date +"%Y %m" | awk '{q=int($2/4);y=$1;if (q==0){q=4;y=y-1;}; printf("%sq%s\n", y, q);}'

A perlsolução seria mais limpo, mas perle DateTimesão um pré-requisito pesado.

#!/usr/bin/perl

use DateTime;

my $today = DateTime->now;
print "today: " . $today->year . "q" . $today->quarter . "\n";

my $ago = DateTime->now->subtract( months=> 4);
print "some time ago: " . $ago->year . "q" . $ago->quarter . "\n"
andcoz
fonte
Legal. Eu gosto da primeira linha date / awk, pois evita chamar (e passar parâmetros para) dateduas vezes.
22412 smokris
4

Divida o formato com data, calcule com awk, formate com printf:

date +"%Y %m" | awk '{printf ("%4dq%1d\n", $1, ($2/4)+1)}'

Apenas data e festança:

echo $(date +%Yq)$(($(date +%m)/4+1))
Usuário desconhecido
fonte
A primeira linha sai 2012q0incorreta.
smokris
@smokris: Você está certo - eu escolhi a linha errada dos meus testes.
usuário desconhecido
4

Uma alternativa, mais como curiosidade. Se o GNU awkestiver envolvido, datenão é necessário:

awk 'BEGIN{print strftime("%Y")"q"int((strftime("%-m")-1)/3)+1}'
homem a trabalhar
fonte
3

Ligue datepara recuperar o ano e o mês atuais e faça o resto com aritmética no shell.

set $(date '+%Y %m');
this_quarter=${1}q$(($2 / 4 + 1))
if [ $2 -eq 1 ]; then
  last_month_quarter=$(($1 - 1))q4
else
  last_month_quarter=${1}q$((($2 - 1) / 4 + 1))
fi
Gilles 'SO- parar de ser mau'
fonte
2

Trimestre do ano deste mês

date +"%Yq$(expr $(expr $(date +%m) - 1) / 3 + 1)"

Trimestre do ano para o mês anterior

date +"%Yq$(expr $(expr $(date -d '-1 month' +%m) - 1) / 3 + 1)"
rthbound
fonte
2

Matemática básica para este trimestre e o trimestre passado:

y1=$(date +%Y)
m1=$(date +%m)
q1=$(( (m1 - 1) / 3 + 1))
y2=$(( y1 - (m1 == 1) ))
m2=$(( (m1 + 10) % 12 + 1 ))
q2=$(( (m2 - 1) / 3 + 1 ))
echo This Quarter: $((y1))q$q1
echo Last Month Quarter: $((y2))q$q2

O script usa as seguintes partes:

  1. $ (unix-cmd) - avalia um comando no shell script
  2. $ ((expr)) - avalia uma expressão matemática
  3. remapear 1..12 -> 1..4 usa a seguinte matemática (m-1) / 3 + 1
  4. avaliar o mês anterior usa módulo de matemática
Stephen Quan
fonte
2

Agora existe o %qformato para mostrar essas informações.

No log de lançamento do coreutils-8.26 de 30 de novembro de 2016:

Novos recursos
... a
data agora aceita o formato% q para produzir o trimestre do ano.

E sim, funciona!

$ date "+%q"
4
$ date "+%Yq%q"
2016q4
fedorqui
fonte
0

Se você quiser o trimestre fiscal de 13 semanas baseado no calendário semanal da ISO, o novo% q útil não funcionará, infelizmente. Aqui está uma versão da solução sem dados do @ manatwork com awk / strftime.

awk 'BEGIN{$x=int((strftime("%V")-1)/13)+1;print strftime("%G")"q"($x>4?4:$x)}'

O último pequeno pedaço ternário lida com os anos da semana bissextos da ISO, onde o último trimestre tem 14 semanas.

TKH
fonte