Можно ли легко отформатировать секунды как удобочитаемое время в bash?
Я не хочу форматировать это как дату, а как количество дней / часов / минут и т. Д ...
Можно ли легко отформатировать секунды как удобочитаемое время в bash?
Я не хочу форматировать это как дату, а как количество дней / часов / минут и т. Д ...
Ответы:
Вы можете использовать что-то вроде этого:
function displaytime {
local T=$1
local D=$((T/60/60/24))
local H=$((T/60/60%24))
local M=$((T/60%60))
local S=$((T%60))
(( $D > 0 )) && printf '%d days ' $D
(( $H > 0 )) && printf '%d hours ' $H
(( $M > 0 )) && printf '%d minutes ' $M
(( $D > 0 || $H > 0 || $M > 0 )) && printf 'and '
printf '%d seconds\n' $S
}
Примеры:
$ displaytime 11617
3 hours 13 minutes and 37 seconds
$ displaytime 42
42 seconds
$ displaytime 666
11 minutes and 6 seconds
Самый простой и понятный способ - это один вкладыш (здесь предполагается, что GNU date
):
Если количество секунд, скажем:
seconds=123456789 # as in one of the answers above
eval "echo $(date -ud "@$seconds" +'$((%s/3600/24)) days %H hours %M minutes %S seconds')"
-> вывод: 1428 days 21 hours 33 minutes 09 seconds
+%T
формат для Часов: Минут: Секунды, чтобы это date +%T "1970-01-01 + ${seconds} seconds"
получить21:33:09
Кредит идет к Стефану Гименесу, но если кто-то хотел бы отображать секунды, только если период меньше минуты, вот моя модифицированная версия, которую я использую (также с фиксированным множественным числом):
converts()
{
local t=$1
local d=$((t/60/60/24))
local h=$((t/60/60%24))
local m=$((t/60%60))
local s=$((t%60))
if [[ $d > 0 ]]; then
[[ $d = 1 ]] && echo -n "$d day " || echo -n "$d days "
fi
if [[ $h > 0 ]]; then
[[ $h = 1 ]] && echo -n "$h hour " || echo -n "$h hours "
fi
if [[ $m > 0 ]]; then
[[ $m = 1 ]] && echo -n "$m minute " || echo -n "$m minutes "
fi
if [[ $d = 0 && $h = 0 && $m = 0 ]]; then
[[ $s = 1 ]] && echo -n "$s second" || echo -n "$s seconds"
fi
echo
}
Альтернативный пример в POSIX:
converts(){
t=$1
d=$((t/60/60/24))
h=$((t/60/60%24))
m=$((t/60%60))
s=$((t%60))
if [ $d -gt 0 ]; then
[ $d = 1 ] && printf "%d day " $d || printf "%d days " $d
fi
if [ $h -gt 0 ]; then
[ $h = 1 ] && printf "%d hour " $h || printf "%d hours " $h
fi
if [ $m -gt 0 ]; then
[ $m = 1 ] && printf "%d minute " $m || printf "%d minutes " $m
fi
if [ $d = 0 ] && [ $h = 0 ] && [ $m = 0 ]; then
[ $s = 1 ] && printf "%d second" $s || printf "%d seconds" $s
fi
printf '\n'
}
convert $s
чтобы получить красивый результат.
Я бы сделал это так:
$ seconds=123456789; echo $((seconds/86400))" days "$(date -d "1970-01-01 + $seconds seconds" "+%H hours %M minutes %S seconds")
1428 days 21 hours 33 minutes 09 seconds
$
Вот один вкладыш выше, разбитый так, чтобы его было легче понять:
$ seconds=123456789
$ echo $((seconds/86400))" days"\
$(date -d "1970-01-01 + $seconds seconds" "+%H hours %M minutes %S seconds")
Выше я повторяю вывод другой команды, которая выполняется внутри $( ... )
подкоманды. Эта подкоманда делает это, вычисляя количество дней (секунд / 86400), затем используя date
команду в другой подкоманде $(date -d ... )
, чтобы сгенерировать часы, минуты и секунды для заданного количества секунд.
Я изменил функцию displaytime выше ... следующим образом:
seconds2time ()
{
T=$1
D=$((T/60/60/24))
H=$((T/60/60%24))
M=$((T/60%60))
S=$((T%60))
if [[ ${D} != 0 ]]
then
printf '%d days %02d:%02d:%02d' $D $H $M $S
else
printf '%02d:%02d:%02d' $H $M $S
fi
}
потому что я всегда хочу видеть ЧЧ: ММ: СС, даже если они нули.
Я строю ответ Атти, который мне понравился как идея.
Вы можете сделать это с помощью встроенной команды bash, printf
которая в качестве аргумента займет секунды с начала эпохи. Не нужно раскошелиться, чтобы бежать date
.
Вы должны установить часовой пояс на UTC, printf
потому что он форматирует время в вашем местном часовом поясе, и вы получите неправильный ответ, если вы не в UTC времени.
$ seconds=123456789
$ TZ=UTC printf "%d days %(%H hours %M minutes %S seconds)T\n" $((seconds/86400)) $seconds
1428 days 21 hours 33 minutes 09 seconds
В моем местном времени (которое в настоящее время NZDT - +1300) ответ будет неправильным, если я не установил часовой пояс
$ seconds=123456789
$ printf "%d days %(%H hours %M minutes %S seconds)T\n" $((seconds/86400)) $seconds
1428 days 09 hours 33 minutes 09 seconds
С и без установки часового пояса
$ seconds=$(( 3600 * 25))
$ printf "%d days %(%H hours %M minutes %S seconds)T\n" $((seconds/86400)) $seconds
1 days 13 hours 00 minutes 00 seconds
$ TZ=UTC printf "%d days %(%H hours %M minutes %S seconds)T\n" $((seconds/86400)) $seconds
1 days 01 hours 00 minutes 00 seconds
printf
ввел это %(datefmt)T
обозначение, начиная с bash-4.2-alpha.
Здесь один
secs=378444
echo $(($secs/86400))d $(($(($secs - $secs/86400*86400))/3600))h:$(($(($secs - $secs/86400*86400))%3600/60))m:$(($(($secs - $secs/86400*86400))%60))s
Выход:
4d 9h:7m:24s
date --date '@1005454800'
дает вам Sun Nov 11 00:00:00 EST 2001
, что составляет 1005454800 секунд после эпохи Unix. Вы можете отформатировать это с опцией date + FORMAT.