养生
serializable(面试官:Java序列化为什么要实现Serializable接口?我懵了)

最近有个公众号粉丝和我聊了聊他面试的经历,一个刚入坑Java两年的新人,由于疫情原因视频面试,而面试官只问了一个问题:“Java序列化为什么要实现Serializable接口?”,结果他一时语塞面试OVER。说实话听到这个问题,我也有些懵逼,平时忙着研究各种中间件、什么高可用框架,可真要回头对Java基础知识较起真,发现自己的技术债欠的太多,所以和大家一起复习一下Java序列化知识。


序列化:Java中的序列化机制能够将一个实例对象信息写入到一个字节流中(只序列化对象的属性值,而不会去序列化方法),序列化后的对象可用于网络传输,或者持久化到数据库、磁盘中。

Java中要使一个类可以序列化,实现java.io.Serializable接口是最简单的。

public class User implements Serializable {    private static final long serialVersionUID = 1L;}

接下来写一个对象信息写入磁盘的例子测试一下:

将User对象信息写入到磁盘当中

@Slf4jpublic class serializeTest {    public static void main(String[] args) throws Exception {        User user = new User();        user.setName("fufu");        user.setAge("18");        serialize(user);        log.info("Java序列化前的结果:{} ", user);        User duser = deserialize();        log.info("Java反序列化的结果:{} ", duser);    }        private static void serialize(User user) throws Exception {        ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream(new File("D:\\111.txt")));        oos.writeObject(user);        oos.close();    }        private static User deserialize() throws Exception {        ObjectInputStream ois = new ObjectInputStream(new FileInputStream(new File("D:\\111.txt")));        return (User) ois.readObject();    }}
序列化前的结果: User(name=fufu, age=18)反序列化后的结果: User(name=fufu, age=18)

这就能解释为什么Java序列化一定要实现Serializable接口了。

    private void writeObject0(Object obj, boolean unshared)        throws IOException    {        boolean oldMode = bout.setBlockDataMode(false);        depth++;        try {            // 省略号。。。。。。。。。。            // remaining cases            if (obj instanceof String) {                writeString((String) obj, unshared);            } else if (cl.isArray()) {                writeArray(obj, desc, unshared);            } else if (obj instanceof Enum) {                writeEnum((Enum<?>) obj, desc, unshared);            } else if (obj instanceof Serializable) {                writeOrdinaryObject(obj, desc, unshared);            } else {                if (extendedDebugInfo) {                    throw new NotSerializableException(                        cl.getName() + "\n" + debugInfoStack.toString());                } else {                    throw new NotSerializableException(cl.getName());                }            }        } finally {            depth--;            bout.setBlockDataMode(oldMode);        }    }

既然已经实现了Serializable接口,为什么还要显示指定serialVersionUID的值呢?

在反序列化时,会根据对象属性自动再生成一个新的serialVersionUID,和序列化时生成的serialVersionUID进行比对,两个serialVersionUID相同则反序列化成功,否则就会抛异常。

序列化对象时如果希望哪个属性不被序列化,则用transient关键字修饰即可

@Datapublic class User implements Serializable {    private transient String name;    private String age;}

一个静态变量不管是否被transient修饰,均不能被序列化。 因为static修饰的属性是属于类,而非对象。

总结

今天就说这么多,如果本文对您有一点帮助,希望能得到您一个点赞哦

[整理了一些Java方面的架构、面试资料(微服务、集群、分布式、中间件等),有需要的小伙伴可以关注公众号【程序员内点事】,无套路自行领取]


面试官:Java序列化为什么要实现Serializable接口?我懵了nerror="javascript:errorimg.call(this);">


顶一下()     踩一下()

热门推荐

发表评论
0评