写一个验证时间格式的正则表达式

今天晚上单位测试问了我这道题。写的挺慢的。正式面试问的话应该已经挂了。
我这里要满足的时间格式 XXXX-XX-XX

分析

先想想闰年有几种情况
  1. 能被4整除
    并且
  2. 当能被100整除的时候,必须能被400整除,所以1600可以,但1500就不行
先看最特殊的闰年-被400整除

用最笨的方法找规律。
0000,0400,0800,1200,1600
2000,2400,2800,3200,3600
4000,4400,4800,5200,5600
得到的规律就是
在第一位是0,2,4,6,8时第二位可以是0,4,8
在第一位是1,3,5,7,9时,第二位可以说2,6
那我们得到的第一种情况的年份的正则表达式应该是
([02468][048]00) | ([13579][26]00)

再来看第二种情况的闰年-能被4整除,且不是100的倍数

还是先找规律
0000,0004,0008,0012,0016
0020,0024,0028,0032,0036
0040,0044,0048,0052,0056
那对应的规律可以说是
前两位随意,第三位是0,2,4,6,8时。对应的第四位0,4,8
第三位是1,3,5时,对应的第四位是2,6
那第二种年份的正则表达式是
([0-9]{2}) (([02468][048]) | [13579][26])

到这里,闰年的情况就讨论完了。

第三种是任意的四个数字 [0-9]{4}

在正则表达式里,这三种情况的排序应该是从最特殊到最一般。也就是上文提到的排序。

然后考虑月份

1,3,5,7,8,10,12月对应的是31天,对应的表达式 (-xx-xx格式)

  • (0[13578]) | (1[02]) - ((0[1-9]) | ([12][0-9]) | (3[01]))
    同样,4,6,9,11月对应的是30天
    -((0[2469]) | (11))- ((0[1-9]) | ([12][0-9]) | (30))
    最后对于二月
    闰年: (-02-((0[1-9]) | ([12][0-9]))
    平年:(-02-((0[1-9]) | (1[0-9] | 2[0-8]))

然后把月份和年份的表达式组合起来
特殊闰年+闰年月日 | 普通闰年 + 闰年月日 | 普通年 + 普通月日
这一步其实很麻烦。会有很多的括号需要调整,感觉半个多小时的时间都浪费在这上头了

得到的代码是这样的(不一定对,简单测一下好像还行)

public class TimePattern {
    public static void main(String[] args ) {
        String pattern =
                "((([02468][048]00)|([13579][26]00))" +
                        "(-02-((0[1-9])|([12][0-9])))|" +
                        "(-((0[13578])|(1[02]))-((0[1-9])|([12][0-9])|(3[01])))|" +
                        "(-(0[469]|11)-((0[1-9])|([12][0-9])|30)))"

                        + "|" +
                "(([0-9]{2})([02468][048]|[13579][26])" +
                        "((-02-((0[1-9])|([12][0-9])))|" +
                        "(-((0[13578])|(1[02]))-((0[1-9])|([12][0-9])|(3[01])))|" +
                        "(-(0[469]|11)-((0[1-9])|([12][0-9])|30))))"

                        + "|" +
                "(([0-9]{4})" +
                        "((-02-((0[1-9])|(([1][0-9])|([2][0-8]))))|" +
                        "(-((0[13578])|(1[02]))-((0[1-9])|([12][0-9])|(3[01])))|" +
                        "(-(0[469]|11)-((0[1-9])|([12][0-9])|30))))";
                       

        System.out.println(Pattern.matches(pattern, "1900-04-31")); //true
        System.out.println(Pattern.matches(pattern, "1900-04-30")); //true
        System.out.println(Pattern.matches(pattern, "1601-02-29")); //false
        System.out.println(Pattern.matches(pattern, "1604-02-29")); //true
        System.out.println(Pattern.matches(pattern, "1900-02-29")); //false
        System.out.println(Pattern.matches(pattern, "0000-02-29")); // true
        System.out.println(Pattern.matches(pattern, "99999-02-28")); //false
        System.out.println(Pattern.matches(pattern, "1421-09-31")); //false
        System.out.println(Pattern.matches(pattern, "1440-02-29")); //true
        System.out.println(Pattern.matches(pattern, "1440-14-29")); //false
    }

你可能感兴趣的:(写一个验证时间格式的正则表达式)