记一次 Could not write JSON: No serializer found for 的坑
今天在返回一个 DTO 实体的时候报错如下:
"message": "Could not write JSON: No serializer found for class com.entity.Question and no properties discovered to create BeanSerializer (to avoid exception, disable SerializationFeature.FAIL_ON_EMPTY_BEANS); nested exception is com.fasterxml.jackson.databind.exc.InvalidDefinitionException: No serializer found for class com.entity.Question and no properties discovered to create BeanSerializer (to avoid exception, disable SerializationFeature.FAIL_ON_EMPTY_BEANS) (through reference chain: com.common.component.RspDTO[\"data\"]->com.dto.common.PageDTO[\"records\"]->java.util.ArrayList[0])",
- 在看到上述问题的时候 首先看到的是 No serializer ,因此首先查看 我们的 PageDTO 类,如下类
public class PageDTO implements Serializable {
private static final long serialVersionUID = -7254888630210798460L;
/**
* 每页显示条数,默认 10
*/
private Integer size;
/**
* 当前页
*/
private Integer current;
private Long total;
/**
* 是否为升序 ASC( 默认: true )
*/
private Boolean isAsc;
private Map<String, String> condition;
private List records = Collections.emptyList();
/**
* get set 方法忽略
*/
}
可以看到 已经实现了 Serializable 接口,并且有正常的 serialVersionUID 字段 ,所以类没有问题
接着继续查看:
com.common.component.RspDTO["data"]->com.dto.common.PageDTO["records"]->java.util.ArrayList[0])",
发现了如下关键字,所以怀疑是 List 中的 entity 的问题,如下:
public class Question extends BaseEntity {
private static final long serialVersionUID = 1L;
private Long id;
private Integer grade;
private Integer subject;
@Version
private Date modifiedAt;
@Override
public Date getModifiedAt() {
return modifiedAt;
}
@Override
public void setModifiedAt(Date modifiedAt) {
this.modifiedAt = modifiedAt;
}
// 忽略 get set 方法
BaseEntity 代码如下:
public class BaseEntity implements Serializable {
private static final long serialVersionUID = 1L;
private Integer isDel;
private Long creatorId;
private Date createdAt;
private Long modifierId;
private Date modifiedAt;
Question 类继承了 BaseEntity 并且实现了 Serializable 接口,查完也没有问题。
那么只能一步步看代码了:
Page page = this.selectPage(new Page(pageDTO.getCurrent(), pageDTO.getSize()), wrapper);
pageDTO.setRecords(page.getRecords());
经查明发现出错的 就是 pageDTO.setRecords(page.getRecords());
然后我们手动获取 page.getRecords(),是没有问题的,那么就剩下了 pageDTO.setRecords() 这个方法,执行后发现果然是这里的问题。
问题来了,为什么一个 setList 的方法会出问题呢?
看了下 page.getRecords() 的内容,发现在 Question 中有一个重写了父类的 modifiedAt 字段,在数据库返回的时候给该值赋值成功,但是他还有个 BaseEntity.modifiedAt 的字段,这个值是 null,就是这里引起的问题
最暴力的方法就是 for 循环 然后给该值赋值,这样就序列化成功了,但是这不是我们想要的效果,之后发现是因为在 setList 的时候,没有使用泛型限定值的类型,所以在set的时候就要全部set进去了。所以一个很小的改动就好:
public class PageDTO <T> implements Serializable{
private List<T> records = Collections.emptyList();
.
.
}
加入限定修饰,然后在使用的时候
Page<Question> page = this.selectPage(new Page(pageDTO.getCurrent(), pageDTO.getSize()), wrapper);
完美解决
遇到坑不怕,怕的是你不深入就跳过去了。