Дата синтаксического анализа в Bash

Как бы вы проанализировали дату в bash, с отдельными полями (годы, месяцы, дни, часы, минуты, секунды) в разные переменные?

Формат даты: YYYY-MM-DD hh:mm:ss

Ответ 1

Должно ли быть bash? Вы можете использовать двоичный файл GNU coreutils /bin/date для многих преобразований:

 $ date --date="2009-01-02 03:04:05" "+%d %B of %Y at %H:%M and %S seconds"
 02 January of 2009 at 03:04 and 05 seconds

Это анализирует данную дату и отображает ее в выбранном формате. Вы можете адаптировать это по своему усмотрению к вашим потребностям.

Ответ 2

Это просто, просто конвертируйте тире и двоеточия в пространство (не нужно менять IFS) и используйте "читать" все в одной строке:

read Y M D h m s <<< ${date//[-:]/ }

Например:

$ date=$(date +'%Y-%m-%d %H:%M:%S')
$ read Y M D h m s <<< ${date//[-: ]/ }
$ echo "Y=$Y, m=$m"
Y=2009, m=57

Ответ 3

У меня был другой формат ввода времени, так что вот более гибкое решение.

Конвертировать даты в BSD/macOS

date -jf in_format [+out_format] in_date

где форматы используют strftime (см. man strftime).

Для заданного формата ввода YYYY-MM-DD hh:mm:ss:

$ date -jf '%Y-%m-%d %H:%M:%S' '2017-05-10 13:40:01'
Wed May 10 13:40:01 PDT 2017

Чтобы прочитать их в отдельные переменные, я использую идею NVRAM, но позволяю вам использовать любой формат strftime:

$ date_in='2017-05-10 13:40:01'

$ format='%Y-%m-%d %H:%M:%S'

$ read -r y m d H M S <<< "$(date -jf "$format" '+%Y %m %d %H %M %S' "$date_in")"

$ for var in y m d H M S; do echo "$var=${!var}"; done
y=2017
m=05
d=10
H=13
M=40
S=01

В скриптах всегда используйте read -r.

В моем случае я хотел выполнить преобразование между /usr/share/zoneinfo имена зон указаны в каталоге /usr/share/zoneinfo):

$ format=%Y-%m-%dT%H:%M:%S%z

$ TZ=UTC date -jf $format +$format 2017-05-10T02:40:01+0200
2017-05-10T00:40:01+0000

$ TZ=America/Los_Angeles date -jf $format +$format 2017-05-10T02:40:01+0200
2017-05-09T17:40:01-0700

Конвертировать даты в GNU/Linux

На Mac вы можете установить версию date GNU как gdate с помощью brew install coreutils.

date [+out_format] -d in_date

где out_format использует strftime (смотрите man strftime).

В команде date GNU coreutils нет способа явно установить формат ввода, так как он пытается определить формат ввода сам по себе, и вещи обычно просто работают. (Подробно, вы можете прочитать руководство в coreutils: Форматы ввода даты.)

Например:

$ date '+%Y %m %d %H %M %S' -d '2017-05-10 13:40:01'
2017 05 10 13 40 01

Чтобы прочитать их в отдельные переменные, измените команду в разделе Mac, заменив команду в скобках на версию GNU.

Чтобы /usr/share/zoneinfo преобразование часовых /usr/share/zoneinfo (см. /usr/share/zoneinfo для имен зон), вы можете указать TZ="America/Los_Angeles" прямо во входной строке. Обратите внимание на буквенное " символы вокруг имени зоны и символ пробела перед in_date:

TZ=out_tz date [+out_format] 'TZ="in_tz" 'in_date

Например:

$ format='%Y-%m-%d %H:%M:%S%z'

$ TZ=America/Los_Angeles date +"$format" -d 'TZ="UTC" 2017-05-10 02:40:01'
2017-05-09 19:40:01-0700

$ TZ=UTC date +"$format" -d 'TZ="America/Los_Angeles" 2017-05-09 19:40:01'
2017-05-10 02:40:01+0000

GNU date также понимает смещения часов для часового пояса:

$ TZ=UTC date +"$format" -d '2017-05-09 19:40:01-0700'
2017-05-10 02:40:01+0000

Ответ 4

$ t='2009-12-03 12:38:15'
$ a=(`echo $t | sed -e 's/[:-]/ /g'`)
$ echo ${a[*]}
2009 12 03 12 38 15
$ echo ${a[3]}
12

Ответ 5

Метод массива, возможно, лучше, но это то, о чем вы конкретно просили:

IFS=" :-"
read year month day hour minute second < <(echo "YYYY-MM-DD hh:mm:ss")

Ответ 6

Pure Bash:

date="2009-12-03 15:35:11"
saveIFS="$IFS"
IFS="- :"
date=($date)
IFS="$saveIFS"
for field in "${date[@]}"
do
    echo $field
done

2009
12
03
15
35
11

Ответ 7

вместо использования сценариев оболочки, включите в свой скрипт себя, как ниже, когда вам нужно:

a=date +%Y 
b=date +%S
c=date +%H

a будет годом b будет секунд c будет часами. и т.д.

Ответ 8

Другое решение проблемы OP:

IFS=' -:' read y m d h m s<<<'2014-03-26 16:36:41'

Преобразование даты в другой формат с помощью BSD date и GNU date:

$ LC_ALL=C date -jf '%a %b %e %H:%M:%S %Z %Y' 'Wed Mar 26 16:36:41 EET 2014' +%F\ %T
2014-03-26 16:36:41
$ gdate -d 'Wed Mar 26 16:36:41 EET 2014' +%F\ %T
2014-03-26 16:36:41

GNU date распознает Wed и Mar даже в неанглийских локалях, но BSD date не делает.

Преобразование секунд с эпохи в дату и время с датой GNU и датой BSD:

$ gdate -d @1234567890 '+%F %T'
2009-02-14 01:31:30
$ date -r 1234567890 '+%F %T'
2009-02-14 01:31:30

Преобразование секунд в часы, минуты и секунды с помощью оболочки POSIX, POSIX awk, GNU date и BSD date:

$ s=12345;printf '%02d:%02d:%02d\n' $((s/3600)) $((s%3600/60)) $((s%60))
05:25:45
$ echo 12345|awk '{printf "%02d:%02d:%02d\n",$0/3600,$0%3600/60,$0%60}'
05:25:45
$ gdate -d @12345 +%T
05:25:45
$ date -r 12345 +%T
05:25:45

Преобразование секунд в дни, часы, минуты и секунды:

$ t=12345678
$ printf '%d:%02d:%02d:%02d\n' $((t/86400)) $((t/3600%24)) $((t/60%60)) $((t%60))
142:21:21:18

Ответ 9

другой чистый bash

$ d="2009-12-03 15:35:11"
$ d=${d//[- :]/|}
$ IFS="|"
$ set -- $d
$ echo $1
2009
$ echo $2
12
$ echo [email protected]
2009 12 03 15 35 11

Ответ 10

Вы пробовали использовать разрез? что-то вроде этого: = день недели date|cut -d" " -f1