Could not write JSON: No serializer found for 的坑

记一次 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);

完美解决


遇到坑不怕,怕的是你不深入就跳过去了。

posted @ 2019-04-23 20:59  haoming1100  阅读(9818)  评论(0编辑  收藏  举报
本文作者: haoming1100 博客链接: http://www.cnblogs.com/haoming1100 版权声明: 本博客所有文章除特别声明外,均采用 CC BY-NC-SA 3.0 许可协议。转载时必须带上原文链接!