How to parse dates in multiple formats using SimpleDateFormat

前端 未结 12 1467
伪装坚强ぢ
伪装坚强ぢ 2020-11-22 08:41

I am trying to parse some dates that are coming out of a document. It would appear users have entered these dates in a similar but not exact format.

here are the for

12条回答
  •  南笙
    南笙 (楼主)
    2020-11-22 09:24

    Matt's approach above is fine, but please be aware that you will run into problems if you use it to differentiate between dates of the format y/M/d and d/M/y. For instance, a formatter initialised with y/M/d will accept a date like 01/01/2009 and give you back a date which is clearly not what you wanted. I fixed the issue as follows, but I have limited time and I'm not happy with the solution for 2 main reasons:

    1. It violates one of Josh Bloch's quidelines, specifically 'don't use exceptions to handle program flow'.
    2. I can see the getDateFormat() method becoming a bit of a nightmare if you needed it to handle lots of other date formats.

    If I had to make something that could handle lots and lots of different date formats and needed to be highly performant, then I think I would use the approach of creating an enum which linked each different date regex to its format. Then use MyEnum.values() to loop through the enum and test with if(myEnum.getPattern().matches(date)) rather than catching a dateformatexception.

    Anway, that being said, the following can handle dates of the formats 'y/M/d' 'y-M-d' 'y M d' 'd/M/y' 'd-M-y' 'd M y' and all other variations of those which include time formats as well:

    import java.text.ParseException;
    import java.text.SimpleDateFormat;
    import java.util.Date;
    
    public class DateUtil {
        private static final String[] timeFormats = {"HH:mm:ss","HH:mm"};
        private static final String[] dateSeparators = {"/","-"," "};
    
        private static final String DMY_FORMAT = "dd{sep}MM{sep}yyyy";
        private static final String YMD_FORMAT = "yyyy{sep}MM{sep}dd";
    
        private static final String ymd_template = "\\d{4}{sep}\\d{2}{sep}\\d{2}.*";
        private static final String dmy_template = "\\d{2}{sep}\\d{2}{sep}\\d{4}.*";
    
        public static Date stringToDate(String input){
        Date date = null;
        String dateFormat = getDateFormat(input);
        if(dateFormat == null){
            throw new IllegalArgumentException("Date is not in an accepted format " + input);
        }
    
        for(String sep : dateSeparators){
            String actualDateFormat = patternForSeparator(dateFormat, sep);
            //try first with the time
            for(String time : timeFormats){
            date = tryParse(input,actualDateFormat + " " + time);
            if(date != null){
                return date;
            }
            }
            //didn't work, try without the time formats
            date = tryParse(input,actualDateFormat);
            if(date != null){
            return date;
            }
        }
    
        return date;
        }
    
        private static String getDateFormat(String date){
        for(String sep : dateSeparators){
            String ymdPattern = patternForSeparator(ymd_template, sep);
            String dmyPattern = patternForSeparator(dmy_template, sep);
            if(date.matches(ymdPattern)){
            return YMD_FORMAT;
            }
            if(date.matches(dmyPattern)){
            return DMY_FORMAT;
            }
        }
        return null;
        }
    
        private static String patternForSeparator(String template, String sep){
        return template.replace("{sep}", sep);
        }
    
        private static Date tryParse(String input, String pattern){
        try{
            return new SimpleDateFormat(pattern).parse(input);
        }
        catch (ParseException e) {}
        return null;
        }
    
    
    }
    

提交回复
热议问题