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);

完美解决


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

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值