Как бы вы проанализировали дату в bash, с отдельными полями (годы, месяцы, дни, часы, минуты, секунды) в разные переменные?
Формат даты: YYYY-MM-DD hh:mm:ss
Как бы вы проанализировали дату в bash, с отдельными полями (годы, месяцы, дни, часы, минуты, секунды) в разные переменные?
Формат даты: YYYY-MM-DD hh:mm:ss
Должно ли быть 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
Это анализирует данную дату и отображает ее в выбранном формате. Вы можете адаптировать это по своему усмотрению к вашим потребностям.
Это просто, просто конвертируйте тире и двоеточия в пространство (не нужно менять 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
У меня был другой формат ввода времени, так что вот более гибкое решение.
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
В моем случае я хотел выполнить преобразование между /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
На 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
$ 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
Метод массива, возможно, лучше, но это то, о чем вы конкретно просили:
IFS=" :-"
read year month day hour minute second < <(echo "YYYY-MM-DD hh:mm:ss")
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
вместо использования сценариев оболочки, включите в свой скрипт себя, как ниже, когда вам нужно:
a=date +%Y
b=date +%S
c=date +%H
a будет годом b будет секунд c будет часами. и т.д.
Другое решение проблемы 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
другой чистый 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
Вы пробовали использовать разрез?
что-то вроде этого:
= день недели date|cut -d" " -f1