惊!未实现Serializable竟让第三方接口回调全军覆没

关注我的公众号:【编程朝花夕拾】,可获取首发内容。

01 引言

前一段时间接收到异常报警,发现传递的类没有实现java.io.Serializable接口导致,调用失败。由于的第三方的统一问题,所以导致调第三方接口回调全军覆没。

从报错信息来,是Dubbo 序列化导致的。

02 问题追踪

从报错的信息中明显的可以看出就是需要实现java.io.Serializable 接口,查看的代码确认后,证实了第三方的jar 包中没有实现java.io.Serializable 接口。

2.1 源码追踪

从报错中可以追溯到:

com.alibaba.com.caucho.hessian.io.SerializerFactory#getDefaultSerializer

主要的方法:

java 复制代码
// cl 是不是 Serializable.class的实现类或者子类
Serializable.class.isAssignableFrom(cl)

isAssignableFrom是用来判断子类和父类的关系的,或者接口的实现类和接口的关系。

所以Dubbo框架会在序列化的时候,校验类是否实现java.io.Serializable。这个是框架行为,并不是所有的框架都会校验。

2.2 Serializable 深究

java.io.Serializable 到底是用来作什么的?

java.io.Serializable 接口是一个标记接口(没有定义任何方法),其主要作用是标识一个类的对象可以被序列化。序列化是指将对象转换为字节流的过程,以便存储到文件、数据库或通过网络传输,之后可以通过反序列化将字节流恢复为对象。

对象实现了java.io.Serializable 表示支持默认的序列化,如果不实现则无法直接序列化,需要借助三方库实现。

java 复制代码
@Data
@AllArgsConstructor
public class User implements Serializable {
    private static final long serialVersionUID = 1L;
    
    String name;
    int age;
}

@Test
void test01() {
    User user = new User("鸣人", 18);
    System.out.println(user);
    try (ByteArrayOutputStream outputStream = new ByteArrayOutputStream();
            ObjectOutputStream oos = new ObjectOutputStream(outputStream)) {
        oos.writeObject(user); // 成功
        oos.flush();

        byte[] bytes = outputStream.toByteArray();
        System.out.println(new String(bytes, StandardCharsets.UTF_8));
    } catch (IOException e) {
        e.printStackTrace();
    }
}

实现Serializable接口执行正常

不实现Serializable接口执行异常

2.3 青睐的框架

java.io.Serializable 受很多流行框架的青睐。如,远程方法调用Dubbo、会话管理Spring Session、消息队列ActiveMQRabbitMQ、缓存持久化Redis、分布式事务Seata等,均需要校验对象是否实现java.io.Serializable 接口。

03 问题解决

找到问题的原因就可以对症下药解决问题了,但是我们遇到的问题是第三方没有实现java.io.Serializable 接口导致的。

本来想着要求对方解决,但是对方以需要开发评估为由婉拒,建议我们兼容处理。

怎么办呢?因为并不是第三方结果返回结果异常,人家是正常返回数据,只不过返回的数据在我们业务系统里传递的时候,导致序列化异常,所以人家并不是着急。

兼容的办法也很简单:直接使用已经实现序列化接口的类代替即可,通用的有java.util.HashMapcom.alibaba.fastjson.JSONObject

04 小结

出现这样的问题,主要就是第三方开发不规范导致的。既然作为提供给别人的SDK,就应该考虑到别人使用的便利性。

反过来讲,我们在和第三方对接时,使用第三方SDK时,里面的实体类能不能拿来直接用,需要我们判断一下,而不能盲目的直接使用。

看似一个小小的问题,平时可能不注意,自己开发的时候要不要实现java.io.Serializable,心里可能都没有谱。一旦发生线上故障,那就是大问题。所以我们要养成良好的编码习惯。

相关推荐
zyk_52014 分钟前
前端渲染pdf文件解决方案-pdf.js
前端·javascript·pdf
Apifox.21 分钟前
Apifox 4月更新|Apifox在线文档支持LLMs.txt、评论支持使用@提及成员、支持为团队配置「IP 允许访问名单」
前端·人工智能·后端·ai·ai编程
划水不带桨27 分钟前
大数据去重
前端
沉迷...32 分钟前
手动实现legend 与 echarts图交互 通过js事件实现图标某项的高亮 显示与隐藏
前端·javascript·echarts
可观测性用观测云1 小时前
观测云数据在Grafana展示的最佳实践
前端
uwvwko1 小时前
ctfhow——web入门214~218(时间盲注开始)
前端·数据库·mysql·ctf
Json____1 小时前
使用vue2开发一个医疗预约挂号平台-前端静态网站项目练习
前端·vue2·网站模板·静态网站·项目练习·挂号系统
nbsaas-boot1 小时前
分布式微服务架构,数据库连接池设计策略
分布式·微服务·架构
littleplayer1 小时前
iOS Swift Redux 架构详解
前端·设计模式·架构
零一码场1 小时前
IMA之ima_read_file 和 ima_post_read_file不同
架构