我知道gawk只有2个函数可以处理日期/时间
mktime和strftime.
所以,我可以使用返回long的mktime来解析任何日期,所以我可以进行任何数学运算,因此我可以使用strftime格式化所需的输出
这就像“1970 01 01 00 00 00”之后任何日期的魅力一样
使用awk,如何在1970年之前格式化日期?
- $awk 'BEGIN{t=mktime("1970 01 01 00 00 00"); print t; print strftime("%Y-%m-%d",t) }'
- 10800
- 1970-01-01
- $awk 'BEGIN{t=mktime("1960 01 01 00 00 00"); print t; print strftime("%Y-%m-%d",t) }'
- -315608400
- awk: cmd. line:1: (FILENAME=- FNR=1) fatal: strftime: second argument less than 0 or too big for time_t
All known POSIX-compliant systems support timestamps from 0 through 2^31 – 1,which is sufficient to represent times through 2038-01-19 03:14:07 UTC. Many systems support a wider range of timestamps,including negative timestamps that represent times before the epoch.
如果给出超出范围的日期,手册没有说明strftime()的作用.
但即使在我的系统上,它确实表现出负面的time_t值,gawk的strftime()函数也不支持它们(虽然mktime()确实如此),因此无法处理1970年之前的日期.我认为这是一个bug在gawk.
(我的建议是使用Perl而不是Awk,但这并不能回答你问的问题.)
原则上,您可以通过在awk中重新实现类似strftime()的函数来重新发明轮子.但这太过分了.
如果您的系统有一个有效的GNU coreutils date命令,您可以从gawk调用它.使用你1960年1月1日的例子:
- $cat 1960.awk
- #!/usr/bin/awk -f
- BEGIN {
- timestamp = mktime("1960 00 00 00 00 00")
- print "mktime() returned " timestamp
- if (0) {
- # This doesn't work
- s = strftime("%Y-%m-%d %H:%M:%S",timestamp)
- print "strftime() returned ",s
- }
- else {
- # This works
- "date '+%Y-%m-%d %H:%M:%S' -d @" timestamp | getline t
- print "The date command printed \"" t "\""
- }
- }
- $./1960.awk
- mktime() returned -318355200
- The date command printed "1959-11-30 00:00:00"
- $
(我放弃了从shell提示符中找出执行此操作所需的引号和反斜杠序列.)
如果你有一个庞大的现有awk程序并且你需要添加这个功能,这可能是有意义的.但是如果你不是坚持用awk做这个,你可能会考虑使用别的东西; awk可能不是你想要完成的工具的正确工具.
或者,如果你真的雄心勃勃,你可以修改gawk来源来正确处理这种情况.