序列化常见问题

举例说明序列化常见问题

1.例子,将OrderTimeCostDTO的对象序列化,再反序列化

@Test
public void m1() throws IOException, ClassNotFoundException {
	OrderTimeCostDTO  otc =OrderTimeCostDTO.builder().cutTime(new Date()).build();
	ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("d:\\stu.txt"));
	oos.writeObject(otc);
	oos.close();
	ObjectInputStream ois = new ObjectInputStream(new FileInputStream("d:\\stu.txt"));
	OrderTimeCostDTO o = (OrderTimeCostDTO)ois.readObject();
	log.info(o+"");
	ois.close();
}

打印结果:

OrderTimeCostDTO(cutTime=Tue Apr 14 15:52:06 CST 2020,...)

2.现在修改OrderTimeCostDTO类,多加一个参数

public class OrderTimeCostDTO implements Serializable{

    /**
     * 多加一个aa参数
     */
    private Date aa;
	
	...
}

3.将d:\stu.txt中的数据反序列化

@Test
public void m1() throws IOException, ClassNotFoundException {
	ObjectInputStream ois = new ObjectInputStream(new FileInputStream("d:\\stu.txt"));
	OrderTimeCostDTO o = (OrderTimeCostDTO)ois.readObject();
	log.info(o+"");
	ois.close();
}

结果:

java.io.InvalidClassException: com.baturu.tms.api.dto.transport.OrderTimeCostDTO; 
local class incompatible: stream classdesc serialVersionUID = -350958361371879346, local class serialVersionUID = -2789788224825224853
at java.io.ObjectStreamClass.initNonProxy(ObjectStreamClass.java:687)
...

解释
将一个对象序列化时,程序会生成一个serialVersionUID
这个serialVersionUID会因为各种原因发生变化,例如类的属性或方法变化,SDK版本的变化等等

所以当我们将类多加一个参数时,serialVersionUID发生了变化,所以将d:\stu.txt文件中的对象对映的serialVersionUID与本地serialVersionUID做比较时
,发现不一致,报错。
在这里插入图片描述

其中model来自反序列化文件时拿到的ObjectStreamClass,osc为反序列化时得到的类的ObjectStreamClass,ObjectStreamClass可以理解为类的描述,包括参数等等,
具体可以看下序列化的源码,推荐:https://blog.csdn.net/u011315960/article/details/89963230

建议手动设置serialVersionUID,兼容类发生的变化,如果没有手动设置serialVersionUID,会根据系统算法默认生成一个serialVersionUID。系统生成的serialVersionUID会因为各种原因发生变化,例如类的属性或方法变化,SDK版本的变化等等,所以加字段很容易反序列化失败。

建议使用JSON传输数据,可读性更强,效率更高

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Mybatis是一种面向对象的持久层框架,用于将Java对象映射到数据库中的表格。在使用Mybatis进行数据库操作时,经常会遇到时间属性的序列化问题。 在Mybatis中,如果实体类中存在时间属性,需要正确配置映射文件中的resultMap节点和sql语句中的时间字段,以便正确地序列化和反序列化时间属性。以下是一些常见的解决方法: 1. 使用Java.util.Date类作为时间类型:在实体类中,将时间属性的类型设置为java.util.Date,然后在映射文件的resultMap节点中将其配置为java.util.Date。这样,Mybatis会默认将时间属性序列化为数据库中的datetime类型。 2. 使用java.sql.Timestamp类作为时间类型:另一种常见的做法是在实体类中使用java.sql.Timestamp类作为时间属性的类型,并在映射文件的resultMap节点中将其配置为java.sql.Timestamp。这样,Mybatis会将时间属性序列化为数据库中的timestamp类型。 3. 使用自定义类型处理器:如果需要将时间属性序列化为其他格式的字符串或从数据库中的字符串反序列化为时间属性,可以使用自定义的类型处理器。通过实现TypeHandler接口,自定义类型处理器可以将时间属性与数据库字段之间进行转换,实现时间序列化的自定义逻辑。 需要注意的是,无论使用哪种方法,都需要在映射文件中正确配置时间字段的名称和类型,以及在sql语句中使用正确的时间字段名称。这样,Mybatis才能正确地序列化和反序列化时间属性。 综上所述,通过正确配置映射文件和使用适当的时间类型,可以解决Mybatis时间序列化问题,确保时间属性在Java对象和数据库字段之间正确地转换。这样,我们就可以在使用Mybatis进行数据库操作时,正确地处理时间属性的持久化和反持久化过程。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值