java序列化(Serializable)

本文深入探讨了对象序列化机制的核心概念,包括如何处理对象引用问题,以及序列化实现过程中的关键步骤。通过具体例子展示了如何利用序列化机制解决对象在磁盘文件中的读写问题,同时介绍了如何在序列化过程中排除某些不希望序列化的字段。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

 

序列化机制只保存对象的类型信息,属性的类型信息和属性值,和方法没有什么关系,你就是给这个类增加10000个方法,序列化内容也不会增加任何东西

 

  简单来说序列化就是一种用来处理对象流的机制,所谓对象流也就是将对象的内容进行流化,流的概念这里不用多说(就是I/O),我们可以对流化后的对象进行读写操作,也可将流化后的对象传输于网络之间(注:要想将对象传输于网络必须进行流化)!在对对象流进行读写操作时会引发一些问题,而序列化机制正是用来解决这些问题的!

问题的引出:

如上所述,读写对象会有什么问题呢?比如:我要将对象写入一个磁盘文件而后再将其读出来会有什么问题吗?别急,其中一个最大的问题就是对象引用!举个例子来说:假如我有两个类,分别是A和B,B类中含有一个指向A类对象的引用,现在我们对两个类进行实例化{ A a = new A(); B b = new B(); },这时在内存中实际上分配了两个空间,一个存储对象a,一个存储对象b,接下来我们想将它们写入到磁盘的一个文件中去,就在写入文件时出现了问题!因为对象b包含对对象a的引用,所以系统会自动的将a的数据复制一份到b中,这样的话当我们从文件中恢复对象时(也就是重新加载到内存中)时,内存分配了三个空间,而对象a同时在内存中存在两份,想一想后果吧,如果我想修改对象a的数据的话,那不是还要搜索它的每一份拷贝来达到对象数据的一致性,这不是我们所希望的!

以下序列化机制的解决方案:

1.保存到磁盘的所有对象都获得一个序列号(1, 2, 3等等)

2.当要保存一个对象时,先检查该对象是否被保存了。

3.如果以前保存过,只需写入"与已经保存的具有序列号x的对象相同"的标记,否则,保存该对象

通过以上的步骤序列化机制解决了对象引用的问题!

序列化的实现

将需要被序列化的类实现Serializable接口,该接口没有需要实现的方法,implements Serializable只是为了标注该对象是可被序列化的,然后使用一个输出流(如:FileOutputStream)来构造一个ObjectOutputStream(对象流)对象,接着,使用ObjectOutputStream对象的writeObject(Object obj)方法就可以将参数为obj的对象写出(即保存其状态),要恢复的话则用输入流。

例子:

import java.io.*;

public class Test {
        public static void main(String[] args) {
                Employee harry = new Employee("Harry Hacker", 50000);
                Manager manager1 = new Manager("Tony Tester", 80000);
                manager1.setSecretary(harry);
                Employee[] staff = new Employee[2];
                staff[0] = harry;
                staff[1] = manager1;
                try {
                        ObjectOutputStream out = new ObjectOutputStream(
                                        new FileOutputStream("employee.dat"));
                        out.writeObject(staff);
                        out.close();
                        ObjectInputStream in = new ObjectInputStream(new FileInputStream(
                                        "employee.dat"));
                        Employee[] newStaff = (Employee[]) in.readObject();
                        in.close();

                        /**
                         * 通过harry对象来加薪 将在secretary上反映出来
                         */
                        newStaff[0].raiseSalary(10);
                        for (int i = 0; i < newStaff.length; i++)
                                System.out.println(newStaff[i]);
                } catch (Exception e) {
                        e.printStackTrace();

                }
        }
}

class Employee implements Serializable {
        public Employee(String n, double s) {
                name = n;
                salary = s;
        }
        /**
         *
         * 加薪水
         *
         */
        public void raiseSalary(double byPercent) {
                double raise = salary * byPercent / 100;
                salary += raise;
        }

        public String toString() {
                return getClass().getName() + "[name = " + name+ ",salary = " + salary+ "]";
        }
        private String name;
        private double salary;
}

class Manager extends Employee

{
        public Manager(String n, double s)
        {
                super(n, s);
                secretary = null;
        }
        /**
         *
         * 设置秘书
         *
         */
        public void setSecretary(Employee s)
        {
           secretary = s;
        }
        
        public String toString()
        {
                return super.toString()+ "[secretary = " + secretary+ "]";
        }

        // secretary代表秘书

        private Employee secretary;

}

修改默认的序列化机制

在序列化的过程中,有些数据字段我们不想将其序列化,对于此类字段我们只需要在定义时给它加上transient关键字即可,对于transient字段序列化机制会跳过不会将其写入文件,当然也不可被恢复。但有时我们想将某一字段序列化,但它在SDK中的定义却是不可序列化的类型,这样的话我们也必须把他标注为transient,可是不能写入又怎么恢复呢?好在序列化机制为包含这种特殊问题的类提供了如下的方法定义:

private void readObject(ObjectInputStream in) throws

IOException, ClassNotFoundException;

private void writeObject(ObjectOutputStream out) throws

IOException;

(注:这些方法定义时必须是私有的,因为不需要你显示调用,序列化机制会自动调用的)

使用以上方法我们可以手动对那些你又想序列化又不可以被序列化的数据字段进行写出和读入操作。

下面是一个典型的例子,java.awt.geom包中的Point2D.Double类就是不可序列化的,因为该类没有实现Serializable接口,在我的例子中将把它当作LabeledPoint类中的一个数据字段,并演示如何将其序列化!

import java.io.*;

import java.awt.geom.*;

public class TransientTest

{

        public static void main(String[] args)
        {
                LabeledPoint label = new LabeledPoint("Book", 5.00, 5.00);
                try
                {
                        System.out.println(label);// 写入前
                        ObjectOutputStream out = new ObjectOutputStream(new
                        FileOutputStream("Label.txt"));
                        out.writeObject(label);
                        out.close();
                        System.out.println(label);// 写入后
                        ObjectInputStream in = new ObjectInputStream(new
                        FileInputStream("Label.txt"));
                        LabeledPoint label1 = (LabeledPoint) in.readObject();
                        in.close();
                        System.out.println(label1);// 读出并加1.0后
                }
                catch (Exception e)
                {
                        e.printStackTrace();
                }
        }
}

class LabeledPoint implements Serializable
{
        public LabeledPoint(String str, double x, double y)
        {
                label = str;
                point = new Point2D.Double(x, y);
        }

        private void writeObject(ObjectOutputStream out) throws IOException
        {
                /**
                 *
                 * 必须通过调用defaultWriteObject()方法来写入
                 *
                 * 对象的描述以及那些可以被序列化的字段
                 *
                 */

                out.defaultWriteObject();
                out.writeDouble(point.getX());
                out.writeDouble(point.getY());
        }

        private void readObject(ObjectInputStream in) throws IOException, ClassNotFoundException
        {
                /**
                 *
                 * 必须调用defaultReadObject()方法
                 *
                 */
                in.defaultReadObject();
                double x = in.readDouble() + 1.0;
                double y = in.readDouble() + 1.0;
                point = new Point2D.Double(x, y);

        }

        public String toString()
        {
                return getClass().getName()+ "[label = " + label+ ", point.getX() = " + point.getX()+ ", point.getY() = " + point.getY()+ "]";
        }
        private String label;
        transient private Point2D.Double point;
}

### Java序列化概念 在Java中,对象的序列化是指把对象转换成字节流的过程,以便于保存到文件或在网络上传输。反序列化则是指将这些字节流转回原始对象的过程。为了使某个类的对象能够被序列化,该类需要实现`java.io.Serializable`接口。 此接口并不提供具体的方法;它只是一个标记接口,表明实现了它的类可以参与序列化进程[^1]。当一个类未实现`Serializable`接口而试图对其进行序列化操作时,将会抛出`NotSerializableException`异常[^2]。 ### Serializable 接口使用方法 为了让一个自定义类支持序列化功能,只需让此类继承`Serializable`接口即可: ```java import java.io.*; class Person implements Serializable { private static final long serialVersionUID = 1L; String name; transient int age; // 不希望age属性被序列化 public Person(String n, int a){ this.name=n; this.age=a; } } ``` 注意,在上述例子中加入了`transient`关键字修饰成员变量`age`,这意味着即使整个类是可序列化的,但是`age`字段不会参与到序列化过程中去。另外还声明了一个名为`serialVersionUID`的静态常量来确保不同版本之间的兼容性。 对于如何实际执行序列化和反序列化过程,则可以通过如下方式完成: #### 序列化对象至文件 ```java public class SerializeDemo { public static void main(String[] args) { Person p = new Person("John", 30); try{ FileOutputStream fileOut = new FileOutputStream("/tmp/person.ser"); ObjectOutputStream out = new ObjectOutputStream(fileOut); out.writeObject(p); out.close(); fileOut.close(); System.out.printf("Serialized data is saved in /tmp/person.ser"); }catch(IOException i){ i.printStackTrace(); } } } ``` #### 从文件读取并恢复对象 ```java public class DeserializeDemo { public static void main(String[] args) { Person p = null; try{ FileInputStream fileIn = new FileInputStream("/tmp/person.ser"); ObjectInputStream in = new ObjectInputStream(fileIn); p = (Person)in.readObject(); in.close(); fileIn.close(); } catch(Exception e){ e.printStackTrace(); return; } System.out.println("Deserialized Person..."); System.out.println("Name: " + p.name); System.out.println("Age: " + p.age); // 这里打印出来的年龄将是默认值0,因为它是瞬态(transient)的 } } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值