考慮最周全的關於日期的正則表達式

一、簡單的日期判斷 (YYYY/MM/DD)

^"d{4}("-|"/|".)"d{1,2}"1"d{1,2}$

二、演化的日期判斷 (YYYY/MM/DD| YY/MM/DD)

^(^("d{4}|"d{2})("-|"/|".)"d{1,2}"3"d{1,2}$)|(^"d{4}年"d{1,2}月"d{1,2}日$)$

三、加入閏年的判斷的

實例:^((((1[6-9]|[2-9]"d)"d{2})-(0?[13578]|1[02])-(0?[1-9]|[12] "d|3[01]))|(((1[6-9]|[2-9]"d)"d{2})-(0?[13456789]|1[012])-(0?[1-9]|[12]"d|30))|(((1[6-9]|[2-9]"d)"d{2})-0?2-(0?[1-9]|1"d|2[0-8]))|(((1[6-9]|[2-9]"d)(0[48]|[2468][048]|[13579][26])|((16|[2468][048]|[3579][26])00))-0?2-29-))$

四、最完整的正則表達式如下:
((^((1[8-9]"d{2})|([2-9]"d{3}))([-"/"._])(10|12|0?[13578])([-"/"._])(3[01]|[12][0-9]|0?[1-9])$)|(^((1[8-9]"d{2})|([2-9]"d{3}))([-"/"._])(11|0?[469])([-"/"._])(30|[12][0-9]|0?[1-9])$)|(^((1[8-9]"d{2})|([2-9]"d{3}))([-"/"._])(0?2)([-"/"._])(2[0-8]|1[0-9]|0?[1-9])$)|(^([2468][048]00)([-"/"._])(0?2)([-"/"._])(29)$)|(^([3579][26]00)([-"/"._])(0?2)([-"/"._])(29)$)|(^([1][89][0][48])([-"/"._])(0?2)([-"/"._])(29)$)|(^([2-9][0-9][0][48])([-"/"._])(0?2)([-"/"._])(29)$)|(^([1][89][2468][048])([-"/"._])(0?2)([-"/"._])(29)$)|(^([2-9][0-9][2468][048])([-"/"._])(0?2)([-"/"._])(29)$)|(^([1][89][13579][26])([-"/"._])(0?2)([-"/"._])(29)$)|(^([2-9][0-9][13579][26])([-"/"._])(0?2)([-"/"._])(29)$))

 

驗證日期的正則表達式加入閏年的判斷以及思路分析

1、什麼是合法的日期範圍?

對於不同的應用場景,這個問題有不同的解釋。這裏採納MSDN中的約定:

DateTime值類型表示值範圍在公元(基督紀元)0001 年 1 月 1 日午夜 12:00:00 到公元 (C.E.) 9999 年 12 月 31 日晚上 11:59:59 之間的日期和時間。

2、關於閏年的闡釋。

關於公曆閏年是這樣規定的:地球繞太陽公轉一週叫做一回歸年,一回歸年長365日5時48分 46秒。因此,公曆規定有平年和閏年,平年一年有365日,比迴歸年短0.2422日,四年共短0.9688日,故每四年增加一日,這一年有366日,就 是閏年。但四年增加一日比四個迴歸年又多0.0312日,400年後將多3.12日,故在400年中少設3個閏年,也就是在400年中只設97個閏年,這 樣公曆年的平均長度與迴歸年就相近似了。由此規定:年份是整百數的必須是400的倍數纔是閏年,例如1900年、2100年就不是閏年。

首先需要驗證年份,顯然,年份範圍爲 0001 - 9999,匹配YYYY的正則表達式爲:

[0-9]{3}[1-9]|[0-9]{2}[1-9][0-9]{1}|[0-9]{1}[1-9][0-9]{2}|[1-9][0-9]{3}

其中 [0-9] 也可以表示爲 "d,但 "d 不如 [0-9] 直觀,因此下面我將一直採用 [0-9]

用正則表達式驗證日期的難點有二:一是大小月份的天數不同,二是閏年的考慮。

對於第一個難點,我們首先不考慮閏年,假設2月份都是28天,這樣,月份和日期可以分成三種情況:

1、月份爲 1, 3, 5, 7, 8, 10, 12,天數範圍爲 01 - 31,匹配MM-DD的正則表達式爲:
(0[13578]|1[02])-(0[1-9]|[12][0-9]|3[01])

2、月份爲 4, 6, 9, 11,天數範圍爲 01-30,匹配MM-DD的正則表達式爲:
(0[469]|11)-(0[1-9]|[12][0-9]|30)

3、月份爲 2,考慮平年情況,匹配MM-DD的正則表達式爲:
02-(0[1-9]|[1][0-9]|2[0-8])

根據上面的成果,我們可以得到匹配平年日期格式爲YYYY-MM-DD的正則表達式:
([0-9]{3}[1-9]|[0-9]{2}[1-9][0-9]{1}|[0-9]{1}[1-9][0-9]{2}|[1-9][0-9]{3})-(((0[13578]|1[02])-(0[1-9]|[12][0-9]|3[01]))|((0[469]|11)-(0[1-9]|[12][0-9]|30))|(02-(0[1-9]|[1][0-9]|2[0-8])))

接着我們來解決第二個難點:閏年的考慮。根據閏年的定義,我們可以將閏年分爲兩類:

1、能被4整除但不能被100整除的年份。尋找後兩位的變化規律,可以很快得到下面的正則匹配:
([0-9]{2})(0[48]|[2468][048]|[13579][26])

2、能被400整除的年份。能被400整除的數肯定能被100整除,因此後兩位肯定是00,我們只要保證前兩位能被4整除即可,相應的正則表達式爲:
(0[48]|[2468][048]|[3579][26])00

好了,所有工作完成後,最強驗證日期的正則表達式也就出來了(加了閏年的驗證)。這個日期正則表達式支持:
YYYY-MM-DD,YYYY/MM/DD,YYYY_MM_DD,YYYY.MM.DD
match : 2008-2-29 2008/02/29
not match : 2008-2-30   2007-2-29

完整的正則表達式如下:
((^((1[8-9]"d{2})|([2-9]"d{3}))([-"/"._])(10|12|0?[13578])([-"/"._])(3[01]|[12][0-9]|0?[1-9])$)|(^((1[8-9]"d{2})|([2-9]"d{3}))([-"/"._])(11|0?[469])([-"/"._])(30|[12][0-9]|0?[1-9])$)|(^((1[8-9]"d{2})|([2-9]"d{3}))([-"/"._])(0?2)([-"/"._])(2[0-8]|1[0-9]|0?[1-9])$)|(^([2468][048]00)([-"/"._])(0?2)([-"/"._])(29)$)|(^([3579][26]00)([-"/"._])(0?2)([-"/"._])(29)$)|(^([1][89][0][48])([-"/"._])(0?2)([-"/"._])(29)$)|(^([2-9][0-9][0][48])([-"/"._])(0?2)([-"/"._])(29)$)|(^([1][89][2468][048])([-"/"._])(0?2)([-"/"._])(29)$)|(^([2-9][0-9][2468][048])([-"/"._])(0?2)([-"/"._])(29)$)|(^([1][89][13579][26])([-"/"._])(0?2)([-"/"._])(29)$)|(^([2-9][0-9][13579][26])([-"/"._])(0?2)([-"/"._])(29)$))

閏年的2月份有29天,因此匹配閏年日期格式爲YYYY-MM-DD的正則表達式爲:
(([0-9]{2})(0[48]|[2468][048]|[13579][26])|((0[48]|[2468][048]| [3579][26])00))-02-29

最後,將平年和閏年的日期驗證表達式合併,我們得到最終的驗證日期格式爲YYYY-MM-DD的正則表達式爲:
(([0-9]{3}[1-9]|[0-9]{2}[1-9][0-9]{1}|[0-9]{1}[1-9][0-9]{2}|[1-9][0-9]{3})-(((0[13578]|1[02])-(0[1-9]|[12][0-9]|3[01]))|((0[469]|11)-(0[1-9]|[12][0-9]|30))|(02-(0[1-9]|[1][0-9]|2[0-8]))))|((([0-9]{2})(0[48]|[2468][048]|[13579][26])|((0[48]|[2468][048]|[3579][26])00))-02-29)

DD/MM/YYYY格式的正則驗證表達式爲:

(((0[1-9]|[12][0-9]|3[01])/((0[13578]|1[02]))|((0[1-9]|[12][0-9]|30)/(0[469]|11))|(0[1-9]|[1][0-9]|2[0-8])/(02))/([0-9]{3}[1-9]|[0-9]{2}[1-9][0-9]{1}|[0-9]{1}[1-9][0-9]{2}|[1-9][0-9]{3}))|(29/02/(([0-9]{2})(0[48]|[2468][048]|[13579][26])|((0[48]|[2468][048]|[3579][26])00)))

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章