玩轉 SHELL 指令碼之:linux date 知多少?
最近好久沒 update 了,一來是近期有點煩人的私事需要處理,二來是工作有點忙,業餘時間還要整個 PPT,蒐集素材啥的,非常耗時間。。。好吧,這都是藉口,其實是人變懶了。。。⊙﹏⊙ 不過我發現最近 1 個月以來,我關注的一些 Blog,一半以上也都沒更新了,看來對大家來說年底都是多事之秋呀~
好了,言歸正傳,之前有介紹過《linux 系統監控、診斷工具之 top 詳解》、《linux 系統監控、診斷工具之 lsof 用法簡介》,今天再來介紹一個使用頻率很高的 linux 命令:date
對日期進行操作,相信每一個 RD 都不會陌生,在我所主要接觸、工作的 3 種語言裡,感覺 shell 下的 date 設計的最簡潔實用,其次是 python,最複雜難用的當屬 java 中的 date 了。
下面要介紹的 date 是指 linux 下的 GNU date, unix 使用者或者非 gnu date 使用者只能 YY 下了,因為下文的絕大多數特性你那都不支援的。
1、獲取當前時間並格式化
june@Win7 192.168.1.101 01:46:27 ~ > date Sat, Dec 21, 2013 1:49:15 AM june@Win7 192.168.1.101 01:49:15 ~ > date -I 2013-12-21 june@Win7 192.168.1.101 02:16:52 ~ > date -d tomorrow +%Y-%m-%d 2013-12-22 june@Win7 192.168.1.101 02:17:49 ~ > date -d yesterday +%Y-%m-%d 2013-12-20 june@Win7 192.168.1.101 02:18:05 ~ > date -d last-day +%Y-%m-%d 2013-12-20 june@Win7 192.168.1.101 02:18:15 ~ > date +'%F %T' 2013-12-21 01:49:26 june@Win7 192.168.1.101 01:49:26 ~ > date +"%Y-%m-%d %H:%M:%S" 2013-12-21 01:49:52 Jun@VAIO 192.168.31.171 23:43:29 ~ > date +%:z # 獲取時區與UTC的差 +08:00 Jun@VAIO 192.168.31.171 23:44:15 ~ >
2、日期的二則運算
GNU date 的日期加減運算是支援自然語言的,主要有三種指令:
- last/next指令
- ago指令
- 負數指令
june@Win7 192.168.1.101 02:10:00 ~ > date -I && date -I -d'last sunday -7 days' 2013-12-21 2013-12-08 june@Win7 192.168.1.101 02:10:03 ~ > # 注意:語法對年月日、時分秒都適用 june@Win7 192.168.1.101 01:59:47 ~ > date -d '20110614 next-day' +%Y-%m-%d 2011-06-15 june@Win7 192.168.1.101 02:01:17 ~ > date -d '20110614 1 days' +%Y-%m-%d 2011-06-15 june@Win7 192.168.1.101 02:01:39 ~ > date -d '20110614 1 days ago' +%Y-%m-%d 2011-06-13 # 注意最好用自然語言,別用 +1 -1 操作,某些早期版本(如RedHat 4U7)不支援可能造成錯誤結果,而且這種寫法會和時區語法衝突,容易出問題。 june@Win7 192.168.1.101 02:01:44 ~ > date -d '20110614 -1 days' +%Y-%m-%d 2011-06-13 june@Win7 192.168.1.101 02:01:57 ~ > date -d '20110614 +1 days' +%Y-%m-%d 2011-06-15 june@Win7 192.168.1.101 02:02:01 ~ >
注意:
- date 自然語言指令在天、小時、分鐘多單位混合的時候注意每個單位都要加限定條件 ago/-,否則預設是 after:
Jun@VAIO 192.168.1.216 15:48:36 ~ >
date -d'-1 hours -30 minutes'
2015年08月 9日 14:18:54
Jun@VAIO 192.168.1.216 15:48:54 ~ >
date -d'1 hours ago 30 mins ago'
2015年08月 9日 14:18:59
Jun@VAIO 192.168.1.216 15:49:00 ~ >
date -d'1 hours 30 mins ago'
2015年08月 9日 16:19:06
Jun@VAIO 192.168.1.216 15:49:06 ~ >
3、字串轉換為日期、指定時間格式
#美國式時間格式
june@Win7 10.59.9.38 18:02:30 ~ >
date -d'31/Oct/2013:00:00:10 +0800' +'%F %T'
date: invalid date `31/Oct/2013:00:00:10 +0800'
june@Win7 10.59.9.38 18:02:34 ~ >
date -d'31 Oct 2013 00:00:10 +0800' +'%F %T'
2013-10-31 00:00:10
june@Win7 10.59.9.38 18:07:34 ~ >
#秒轉為分
date -u -d@3613 +%H"h"%M"min"%S"s"
01h00min13s
#日期轉時間戳
june@Win7 192.168.1.101 02:02:01 ~ >
date -d "2010-12-11" +%s
1291996800
#時間戳轉日期
june@Win7 192.168.1.101 02:04:22 ~ >
date -d@1291996800 -I
2010-12-11
# 下面的是非常規方法,拓展思路而已
june@Win7 192.168.1.101 02:10:03 ~ >
echo 1307980800 | awk '{T=strftime("%F %T",$1);print T}'
2011-06-14 00:00:00
june@Win7 192.168.1.101 02:16:15 ~ >
echo '1307980800' |sed -r -e "s/(.*)/date -d @1 '+%Y-%m-%d %H:%M:%S'/e"
2011-06-14 00:00:00
june@Win7 192.168.1.101 02:16:52 ~ >
#此方法在大於2038年的時候會有問題,即使是 64bit 也是有問題的。32473710849
date -d "UTC 1970-01-01 1234567890 secs"
4、注意時區問題
date 裡面時間的 + - 注意格式會影響結果/時區(最安全的寫法是不要帶符號,用自然語言即可):
date -d'2013-07-01 09:52:33 +1 minutes' # 這個+1被當成時區了
Mon, Jul 01, 2013 4:53:33 PM
date -d'2013-07-01 09:52:33 1 minutes' # 同樣 -1 也會有問題
Mon, Jul 01, 2013 9:53:33 AM
date -u --date='+2 minutes 13-07-01 09:52:33' # 把 + - 時間放在最前面也行
Mon Jul 1 09:54:33 UTC 2013
date -u --date='13-07-01 09:52:33 +0 +2 minutes' # 注意前面的 -u UTC時間,少了也會有問題
Mon Jul 1 09:54:33 UTC 2013
date -u --date='13-07-01 09:52:33 GMT +2 minutes' # 指定時區
Mon Jul 1 09:54:33 UTC 2013
5、最後的技能:
(1)當 %H%M 遇上 [[ ]]
請看下面這段程式碼,很多人估計會犯的錯誤:
[[ `date +'%H%M'` -eq 0 ]]
恭喜你,你每天早上的 8、9 點執行指令碼的話都會遇到:
value too great for base (error token is "0950") 這樣的錯誤,
這是因為 [[]] 把 0950 當成 8 進位制來解析了,所以拋異常了。
這裡的規則是:
以 0 開頭預設 8 進位制,以 0x 開頭 16 進位制,或者你用 n# 的方式手動指定:
[[ 0010 -eq 8 ]] && echo 111
111
[[ 8#10 -eq 8 ]] && echo 111
111
[[ 0x10 -eq 16 ]] && echo 111
111
解決辦法有如下 4 種,可以選擇你認為最爽的一種修復:
- (1.1)換成 [ 0 -eq 0950 ] && echo 1 即可
- (1.2)或者手動指定進位制也行 [[ 10#`date +%H%M` -eq 10#0950 ]] && echo 1
- (1.3)兩邊當做字串來對待 [[ "`date +'%H%M'`" == "0950" ]] && echo 1
- (1.4)加 - 格式化佔位符,讓 '%H%M' 這種格式不帶前導符 0:
june@Win7 192.168.1.101 02:30:03 ~ >
date +'[%H%M]'
[0230]
june@Win7 192.168.1.101 02:30:19 ~ >
date +'[%_H%_M]'
[ 230]
june@Win7 192.168.1.101 02:30:26 ~ >
date +'[%-H%-M]'
[230]
june@Win7 192.168.1.101 02:30:35 ~ >
(2)如何快速獲取一個日期列表
june@Win7 192.168.1.101 02:30:35 ~ ># seq -w,seq -f%.8g, echo {..}
date -f <(seq -f%.0f 20130227 20130301) +%Y%m%d 2>/dev/null
20130227
20130228
20130301
june@Win7 192.168.1.101 02:33:50 ~ >
(3)如何獲取任意日期的月末/月初(第一天/最後一天)?
判斷今天是不是月末:
[ `date --date='next day' +'%B'` != `date +'%B'` ] && echo 'end of month' || echo 'not end of month'
[[ `date +%d` -eq `echo $(cal)|grep -Po 'd+$'` ]] && echo 月末 || echo 非月末
得到當月/上月的第1天/最後一天:
# First Day, current month: ## %d = day of month.
date -d "-0 month -$(($(date +%d)-1)) days"
# First Day, last month:
date -d "-1 month -$(($(date +%d)-1)) days"
# Last Day, last month:
date -d "-$(date +%d) days -0 month"
# Last Day, current month:
date -d "-$(date +%d) days +1 month"
# Last Day, month before last month:
date -d "-$(date +%d) days -1 month"
得到指定日期當月的最後一天:
Jun@VAIO 192.168.1.216 23:57:02 ~ >
getlastDay_func1(){ echo `echo $(cal $(echo $1|awk -vFIELDWIDTHS="4 2 2" '{print $2,$1}'))|grep -Po 'd+$'`; }
Jun@VAIO 192.168.1.216 23:57:45 ~ >
getlastDay_func1 20150214
28
Jun@VAIO 192.168.1.216 23:56:44 ~ >
getlastDay_func2(){ echo `date -d"-1 days +1 month $(echo $1|sed -r 's/(....)(..).*/1-2-01/')" +%Y%m%d`; }
Jun@VAIO 192.168.1.216 23:56:52 ~ >
getlastDay_func2 20150214
20150228
Jun@VAIO 192.168.1.216 23:57:02 ~ >
好吧,今天的主題就到此為止了,希望本文能對你有所幫助。
簡單、高效一直是每個 Linuxer 追求的目標,如果你也有這方面的技巧/問題需要交流,歡迎通過下面的評論來與我保持聯絡。
6、Ref
(1)一個關於date -d '1 month ago' 的 “bug”
http://hi.baidu.com/leejun_2005/item/8cb255757225835b0d0a078a
(2)date非常規用法總結
http://scmbob.org/special_usage_of_date.html
(3)shell 轉換時間戳獲取兩個時間段的所有日期,返回list
http://hi.baidu.com/leejun_2005/item/92854594db7ca831326eeb69
(4)python datetime 時間日期處理小結
http://hi.baidu.com/leejun_2005/item/abaf4d0e3aac8a153b53ee69
(5)自定義 java 日期、時間 處理函式集