java8中新的时间类LocalDate、LocalTime、LocalDateTime在进行前后端交互和数据持久化时会遇到格式问题和映射问题,总结如下:
1、使用springboot jpa时报错:数据库操作失败,提示【could not deserialize】(无法反序列化)
解决方法:pom文件引入下面的jar包
<dependency>
<groupId>org.hibernate</groupId>
<artifactId>hibernate-java8</artifactId>
<version>5.1.2.Final</version>
</dependency>
2、上面jar包引入后就没再报错,但前端时间字段显示格式不是我们常见的“yyyy-MM-dd hh:mm:ss”,而是下面的样子:
"createTime": {
"month": "DECEMBER",
"year": 2018,
"dayOfMonth": 12,
"dayOfWeek": "WEDNESDAY",
"dayOfYear": 346,
"monthValue": 12,
"hour": 10,
"minute": 22,
"nano": 0,
"second": 25,
"chronology": {
"id": "ISO",
"calendarType": "iso8601"
}
解决方法:
引入下面的jar包
<dependency>
<groupId>com.fasterxml.jackson.datatype</groupId>
<artifactId>jackson-datatype-jsr310</artifactId>
<version>2.8.5</version>
</dependency>
同时在类的时间字段上加上注解:@JsonFormat(pattern = "yyyy-MM-dd hh:mm:ss")
@JsonFormat(pattern = "yyyy-MM-dd hh:mm:ss")
private LocalDateTime createTime;
@JsonFormat(pattern = "yyyy-MM-dd hh:mm:ss")
private LocalDateTime updateTime;
现在再看一下返回结果
"createTime": "2018-12-11 06:41:33",
"updateTime": "2018-12-11 06:41:33",
格式完全正确了
3、虽然格式现在已经正确了,但是还有一个小问题:返回结果是12小时制,数据库中是24小时制,一般我们也是用24小时制
解决方法:
在@JsonFormat注解中pattern = "yyyy-MM-dd HH:mm:ss",就是把小写的hh换成HH就可以了
@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss",timezone="GMT+8")
private LocalDateTime createTime;
@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss",timezone="GMT+8")
private LocalDateTime updateTime;
返回结果如下:
"createTime": "2018-12-11 18:41:33",
"updateTime": "2018-12-11 18:41:33",
完美解决。
4、我是用的持久化框架是JPA JPA包装的是hibernate,所以第一步用的是hibernate-java8,如果是mybatis的话,请引入下面的jar包
<dependency>
<groupId>org.mybatis</groupId>
<artifactId>mybatis-typehandlers-jsr310</artifactId>
<version>1.0.2</version>
</dependency>
5、实际设计中,create_time和update_time这俩字段一般是数据库自动生成,这个时候如果使用JPA的话,需要在相应属性加上注解:@Column(insertable = false, updatable = false)。如下所示:
@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss",timezone="GMT+8")
@Column(insertable = false, updatable = false)
private LocalDateTime createTime;
/**更新时间*/
@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss",timezone="GMT+8")
@Column(insertable = false, updatable = false)
private LocalDateTime updateTime;
这样的话我们子新增和更新表数据的时候,这两个字段会使用数据库自动生成和更新。
6、前端在RequestBody中传时间参数,后端用LocalDate、LocalTime、LocalDateTime来接收的话,直接在字段上使用@JsonFormat注解,如下面的operateTime字段。
@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss",timezone="GMT+8")
private LocalDateTime operateTime;
注意前端传的数据一定要严格符合pattern中指定的格式,如果前端传"2019-02-13",而pattern中的格式为"yyyy-MM-dd HH:mm:ss"的话,也会报错,不能自动转成"2019-02-13 00:00:00"。
7、前端在RequestParam和PathVariable中传时间参数的话,上面的方法就不好使了,这时候需要加一个日期转换器,使用Spring中的Converter来实现
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.core.convert.converter.Converter;
import java.time.LocalDate;
import java.time.LocalDateTime;
import java.time.format.DateTimeFormatter;
@Configuration
public class DateConfig {
@Bean
public Converter<String, LocalDate> localDateConverter() {
return new Converter<String, LocalDate>() {
@Override
public LocalDate convert(String source) {
return LocalDate.parse(source, DateTimeFormatter.ofPattern("yyyy-MM-dd"));
}
};
}
@Bean
public Converter<String, LocalDateTime> localDateTimeConverter() {
return new Converter<String, LocalDateTime>() {
@Override
public LocalDateTime convert(String source) {
return LocalDateTime.parse(source, DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss"));
}
};
}
}
把上面代码加入到项目中就可以。