使用正则表达式判断日期字符串格式是否合法遇到的问题(解决)
引言
我们在使用SpringMVC从前端接受传递过来的日期数据时,默认传递过来的数据是String类型,如果我们从前端传递过来的数据格式是yyyy/MM/dd,SpringMVC有内置类型转化器会将String类型自动转化成Date类型。但如果我们从前端传递过来的数据格式是yyyy-MM-dd,SpringMVC的内置转化器就不能自动转化了,就得我们自定义一个类型转化器。
注意:如果我们自定义类型转化器之后,SpringMVC支持内置类型转化器就不会在生效了。
创建自定义字符串类型转日期类型的类型转化器类
package com.qrj.converter;
import org.springframework.core.convert.converter.Converter;
import org.springframework.core.convert.support.GenericConversionService;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.regex.Pattern;
/**
* @author: qrj
* @description 自定义字符串类型转日期类型的类型转换器
* @Date: 2023/9/29 16:41
*/
public class StringToDateConverter implements Converter<String, Date> {
@Override
public Date convert(String s) {
Date date = null;
try{
SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd");
date = simpleDateFormat.parse(s);
}catch (Exception e){
e.printStackTrace();
}
return date;
}
}
我们一个在类型转化器类里面写一个main方法进行测试,看能不能将yyyy-MM-dd格式的字符串转化成Date类型的数据。
public static void main(String[] args) {
//创建类型转换服务实例
GenericConversionService genericConversionService = new GenericConversionService();
//想类型转换服务实例中注册自定义类型转换器
genericConversionService.addConverter(new StringToDateConverter());
//使用自定义类型转换器
String dateStr = "2023-9-29";
Date date = genericConversionService.convert(dateStr,Date.class);
System.out.println(date);
}
测试结果如下:
测试成功,自定义类型转化器可以将yyyy-MM-dd格式的字符串转化成Date类型的数据。
但问题来咯!有自定义类型转化器之后,SpringMVC支持内置类型转化器就不会在生效。如果我们前端数据传一个yyyy/MM/dd格式的日期呢???
解决这个问题,我们是不是应该进行一个判断,这里正则表达式就出场咯。如果前端传递过来的是yyyy-MM-dd格式的字符串,就将它转化为Date类型,如果前端传递过来的是yyyy/MM/dd格式的字符串,也将它转化为Date类型。
用正则表达式判断日期字符串是否合法
package com.qrj.converter;
import org.springframework.core.convert.converter.Converter;
import org.springframework.core.convert.support.GenericConversionService;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.regex.Pattern;
/**
* @author: qrj
* @description 自定义字符串类型转日期类型的类型转换器
* @Date: 2023/9/29 16:41
*/
public class StringToDateConverter implements Converter<String, Date> {
@Override
public Date convert(String s) {
Date date = null;
SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd");
try{
//可以使用正则表达式判断日期字符串格式是否合法(支持 yyyy-MM-dd 和 yyyy/MM/dd 格式)
boolean isMate1 = Pattern.matches("^(\\d{4})-(\\d{2})-(\\d{2})$",s);
if (isMate1 == true){
simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd");
}
boolean isMate2 = Pattern.matches("^(\\d{4})/(\\d{2})/(\\d{2})$",s);
if (isMate2 == true){
simpleDateFormat = new SimpleDateFormat("yyyy/MM/dd");
}
date = simpleDateFormat.parse(s);
}catch (Exception e){
e.printStackTrace();
}
return date;
}
}
注意:上面Pattern.matches()方法是Java中用于检查给定字符串是否与给定模式匹配的静态方法。它接受两个参数:第一个参数是表示正则表达式模式的字符串,第二个参数是待检查的字符串。
编写测试代码,进行测试:
//测试自定义类型转换器
public static void main(String[] args) {
//创建类型转换服务实例
GenericConversionService genericConversionService = new GenericConversionService();
//想类型转换服务实例中注册自定义类型转换器
genericConversionService.addConverter(new StringToDateConverter());
//使用自定义类型转换器
String dateStr = "2023/9/29";
Date date = genericConversionService.convert(dateStr,Date.class);
System.out.println(date);
}
测试结果如下:
报错原因是无法解析的date :"2023/9/29"
遇到问题
在网上查询资料得知,原因是在 SimpleDateFormat中使用的日期格式与实际的日期字符串格式不匹配,导致无法解析。在"2023/9/29"这种格式的日期字符串中,月份部分只有一位数字,日期部分有两位数字,而在正则表达式中(\\d{2})日期和月份部分匹配两位数字。这使得2023/9/29与正则表达式不匹配,返回false,无法将2023/9/29这种格式的日期字符串解析。
解决问题
为了解决这一问题,可以通过修改正则表达式的月份跟日期部分为(\\d{1,2})使其匹配一到两位数字。以下是修改后的代码:
package com.qrj.converter;
import org.springframework.core.convert.converter.Converter;
import org.springframework.core.convert.support.GenericConversionService;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.regex.Pattern;
/**
* @author: qrj
* @description 自定义字符串类型转日期类型的类型转换器
* @Date: 2023/9/29 16:41
*/
public class StringToDateConverter implements Converter<String, Date> {
@Override
public Date convert(String s) {
Date date = null;
SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd");
try{
//可以使用正则表达式判断日期字符串格式是否合法(支持 yyyy-MM-dd 和 yyyy/MM/dd 格式)
boolean isMate1 = Pattern.matches("^(\\d{4})-(\\d{2})-(\\d{2})$",s);
if (isMate1){
simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd");
}
boolean isMate2 = Pattern.matches("^(\\d{4})/(\\d{1,2})/(\\d{1,2})$",s);
if (isMate2){
simpleDateFormat = new SimpleDateFormat("yyyy/MM/dd");
}
date = simpleDateFormat.parse(s);
}catch (Exception e){
e.printStackTrace();
}
return date;
}
//测试自定义类型转换器
public static void main(String[] args) {
//创建类型转换服务实例
GenericConversionService genericConversionService = new GenericConversionService();
//想类型转换服务实例中注册自定义类型转换器
genericConversionService.addConverter(new StringToDateConverter());
//使用自定义类型转换器
String dateStr = "2023/9/29";
String dateStr2 = "2023-9-30";
Date date = genericConversionService.convert(dateStr,Date.class);
Date date2 = genericConversionService.convert(dateStr2,Date.class);
System.out.println(date);
System.out.println(date2);
}
}
进行测试如下:
成功将yyyy-MM-dd格式和yyyy/MM/dd格式的字符串转化成Date类型的数据。
正则表达式
"^(\\d{4})-(\\d{2})-(\\d{2})$"
"^(\\d{4})/(\\d{1,2})/(\\d{1,2})$"
^:匹配字符串的开头
(\\d{4}):匹配四位数字,表示年份
-或/:匹配横杠或斜杠字符
(\\d{1,2}):匹配一到两位数字,表示月份或日期
$:匹配字符串的结尾