对象序列化与反序列化

分手后的思念是犯贱 2022-03-20 11:28 426阅读 0赞

对象序列化

哈喽大家好这里是狗蛋子,今天让我们讨论一下对象序列化的知识体系。
我现在就是想问大家一个问题,大家众所周知,new出来的对象是在内存中存放着,但是假如我现在要通过网络传输给你,我该怎么做?这样说未免太生涩了,我举个例子理解一下:假如你在一家餐馆,你在前台点菜了,老板通过网络将你想吃的菜肴发送给了后厨,后厨就能将你的饭菜做好,你知道这怎么实现的么?不知道就对啦! 当然你知道肯定是更好哈哈哈哈,好了不开玩笑,今天就和大家探讨一下这个问题:


序列化和反序列化

序列是将对象的状态信息转化为可以存储或者可以网络传输的形式的过程。一般将一个对象存储到一个储存媒介,例如档案或记忆体缓冲等,在网络传输过程中,可以是字节或者XML等格式;而字节或者XML格式的可以还原成完全相等的对象,这个相反的过程又称为反序列化。


首先,java是支持这个事件发生的,意思就是说,java有这样一个功能,首先我们要了解,实现功能一般都是有一个类或者一个接口能完成这项任务,这里我们就需要实现一个接口,这里的接口叫做:java.io.Serializable,Java类通过实现java.io.Serialization接口来启用序列化功能,未实现此接口的类将无法将其任何状态或者信息进行序列化或者反序列化。可序列化类的所有子类型都是可以序列化的。序列化接口没有方法或者字段,仅用于标识可序列化的语义。只要你实现这个接口,JVM就会将你的信息序列化,你就可以将信息通过网络传输,然后到了终点站以后再反序列化还原你发的信息,这样就可以将内存中的实例通过网络传输了。当试图对一个对象进行序列化时,如果遇到一个没有实现java.io.Serialization接口的对象时,将抛出NotSerializationException异常。


以前会用上面的方式进行传输,现在能力升级了,可以通过json就可以实现这个功能,现在会搭建Rest服务,在网际之间传输json,然后设计成Restful接口来完成这件事。聊天传输的一条条消息不是一条条实例,而是JavaScript对象,叫做json,现在都用FastJson这个阿里巴巴开发的组件完成。当然,以后随着深入学习,你会了解这个的,现在只是普及一下这个知识点。


但是我们还是得继续学习一下这个序列化和反序列化的过程
首先我们创建一个Student类,学生有ID,name等参数,然后写好构造函数还有get()set()方法:

  1. public class Stu implements java.io.Serializable {
  2. private static final long serialVersionUID = 8918676416282216936L;
  3. private int id;
  4. private String name;
  5. public Stu(int id, String name) {
  6. this.id = id;
  7. this.name = name;
  8. }
  9. public Stu(){ }
  10. public int getId() {
  11. return id;
  12. }
  13. public void setId(int id) {
  14. this.id = id;
  15. }
  16. public String getName() {
  17. return name;
  18. }
  19. public void setName(String name) {
  20. this.name = name;
  21. }
  22. }

之后新创建一个类TestCase用于模拟序列化和反序列化:

  1. public class TestCase {
  2. public static void main(String[] args) throws Exception{
  3. Stu stu = new Stu(2,"jack");
  4. ObjectOutputStream out = new ObjectOutputStream(new FileOutputStream("src/com/hzy/IO/data/stu.data"));
  5. out.writeObject(stu);
  6. Stu stu2 = new Stu();
  7. ObjectInputStream in = new ObjectInputStream(new FileInputStream("src/com/hzy/IO/data/stu.data"));
  8. stu2 =(Stu)in.readObject();
  9. System.out.println(stu2);
  10. }
  11. }

我来解释一下代码,首先大家看,这里的输出流不再是FileOutputStream了。而是ObjectOutputStream,这个可是个高级流——对象输出流。相当于你把这个Jack同学写进指定目录里,然后又声明一个stu2,里面什么成员变量也没写,然后用一个ObjectInputStream目录里面那个已经序列化的文件——stu.data,然后把读到的对象转化成stu2,然后再把stu2输出,这时,stu和stu2表示的对象是一样的这样就反序列化了。
在这里插入图片描述


这里再说一个关键字:transient,是这样使用的:

  1. private int id;
  2. private transient String name;

因为有这个关键字,所以GC回收会非常快,所以有这个关键字的成员变量就不参与序列化。可以控制需要序列化的成员变量有哪些。


发表评论

表情:
评论列表 (有 0 条评论,426人围观)

还没有评论,来说两句吧...

相关阅读

    相关 Java对象序列序列

    一、序列化和反序列化介绍 在Java中,对象的序列化与反序列化被广泛应用到RMI(远程方法调用)及网络传输中。 序列化:指将Java对象数据保存到磁盘文件中或者传递给其

    相关 序列序列

    因为TCP/IP协议只支持字节数组的传输,不能直接传对象。对象序列化的结果一定是字节数组!当两个进程在进行远程通信时,彼此可以发送各种类型的数据。无论是何种类型的数据,都会以二