菜鸟教程 -- 学的不仅是技术,更是梦想!

Java 教程
(追記) (追記ここまで)

Java 序列化

Java 序列化是一种将对象转换为字节流的过程,以便可以将对象保存到磁盘上,将其传输到网络上,或者将其存储在内存中,以后再进行反序列化,将字节流重新转换为对象。

序列化在 Java 中是通过 java.io.Serializable 接口来实现的,该接口没有任何方法,只是一个标记接口,用于标识类可以被序列化。

当你序列化对象时,你把它包装成一个特殊文件,可以保存、传输或存储。反序列化则是打开这个文件,读取序列化的数据,然后将其还原为对象,以便在程序中使用。

序列化是一种用于保存、传输和还原对象的方法,它使得对象可以在不同的计算机之间移动和共享,这对于分布式系统、数据存储和跨平台通信非常有用。

以下是 Java 序列化的基本概念和用法:

实现 Serializable 接口: 要使一个类可序列化,需要让该类实现 java.io.Serializable 接口,这告诉 Java 编译器这个类可以被序列化,例如:

实例

import java.io.Serializable;

public class MyClass implements Serializable {
// 类的成员和方法
}

序列化对象: 使用 ObjectOutputStream 类来将对象序列化为字节流,以下是一个简单的实例:

实例

MyClass obj = new MyClass();
try {
FileOutputStream fileOut = new FileOutputStream("object.ser");
ObjectOutputStream out = new ObjectOutputStream(fileOut);
out.writeObject(obj);
out.close();
fileOut.close();
} catch (IOException e) {
e.printStackTrace();
}

上述代码将一个名为 "object.ser" 文件中的 obj 对象序列化。

反序列化对象: 使用 ObjectInputStream 类来从字节流中反序列化对象,以下是一个简单的实例:

实例

MyClass obj = null;
try {
FileInputStream fileIn = new FileInputStream("object.ser");
ObjectInputStream in = new ObjectInputStream(fileIn);
obj = (MyClass) in.readObject();
in.close();
fileIn.close();
} catch (IOException e) {
e.printStackTrace();
} catch (ClassNotFoundException e) {
e.printStackTrace();
}

上述代码从 "object.ser" 文件中读取字节流并将其反序列化为一个 MyClass 对象。

类 ObjectInputStream 和 ObjectOutputStream 是高层次的数据流,它们包含反序列化和序列化对象的方法。

ObjectOutputStream 类包含很多写方法来写各种数据类型,但是一个特别的方法例外:

publicfinalvoidwriteObject(Objectx)throwsIOException

上面的方法序列化一个对象,并将它发送到输出流。相似的 ObjectInputStream 类包含如下反序列化一个对象的方法:

publicfinalObjectreadObject()throwsIOException, ClassNotFoundException

该方法从流中取出下一个对象,并将对象反序列化。它的返回值为Object,因此,你需要将它转换成合适的数据类型。

实例

为了演示序列化在 Java 中是怎样工作的,我将使用之前教程中提到的 Employee 类,假设我们定义了如下的 Employee 类,该类实现了Serializable 接口。

Employee.java 文件代码:

publicclassEmployeeimplementsjava.io.Serializable{publicStringname; publicStringaddress; publictransientintSSN; publicintnumber; publicvoidmailCheck(){System.out.println("Mailing a check to " + name + "" + address); }}

请注意,一个类的对象要想序列化成功,必须满足两个条件:

该类必须实现 java.io.Serializable 接口。

该类的所有属性必须是可序列化的。如果有一个属性不是可序列化的,则该属性必须注明是短暂的。

如果你想知道一个 Java 标准类是否是可序列化的,请查看该类的文档。检验一个类的实例是否能序列化十分简单, 只需要查看该类有没有实现 java.io.Serializable接口。


序列化对象

ObjectOutputStream 类用来序列化一个对象,如下的 SerializeDemo 例子实例化了一个 Employee 对象,并将该对象序列化到一个文件中。

该程序执行后,就创建了一个名为 employee.ser 文件。该程序没有任何输出,但是你可以通过代码研读来理解程序的作用。

注意: 当序列化一个对象到文件时, 按照 Java 的标准约定是给文件一个 .ser 扩展名。

SerializeDemo.java 文件代码:

importjava.io.*; publicclassSerializeDemo{publicstaticvoidmain(String[]args){Employeee = newEmployee(); e.name = "Reyan Ali"; e.address = "Phokka Kuan, Ambehta Peer"; e.SSN = 11122333; e.number = 101; try{FileOutputStreamfileOut = newFileOutputStream("/tmp/employee.ser"); ObjectOutputStreamout = newObjectOutputStream(fileOut); out.writeObject(e); out.close(); fileOut.close(); System.out.printf("Serialized data is saved in /tmp/employee.ser"); }catch(IOExceptioni){i.printStackTrace(); }}}

反序列化对象

下面的 DeserializeDemo 程序实例了反序列化,/tmp/employee.ser 存储了 Employee 对象。

DeserializeDemo.java 文件代码:

importjava.io.*; publicclassDeserializeDemo{publicstaticvoidmain(String[]args){Employeee = null; try{FileInputStreamfileIn = newFileInputStream("/tmp/employee.ser"); ObjectInputStreamin = newObjectInputStream(fileIn); e = (Employee)in.readObject(); in.close(); fileIn.close(); }catch(IOExceptioni){i.printStackTrace(); return; }catch(ClassNotFoundExceptionc){System.out.println("Employee class not found"); c.printStackTrace(); return; }System.out.println("Deserialized Employee..."); System.out.println("Name: " + e.name); System.out.println("Address: " + e.address); System.out.println("SSN: " + e.SSN); System.out.println("Number: " + e.number); }}

以上程序编译运行结果如下所示:

Deserialized Employee...
Name: Reyan Ali
Address:Phokka Kuan, Ambehta Peer
SSN: 0
Number:101

这里要注意以下要点:

readObject() 方法中的 try/catch代码块尝试捕获 ClassNotFoundException 异常。对于 JVM 可以反序列化对象,它必须是能够找到字节码的类。如果JVM在反序列化对象的过程中找不到该类,则抛出一个 ClassNotFoundException 异常。

注意,readObject() 方法的返回值被转化成 Employee 引用。

当对象被序列化时,属性 SSN 的值为 111222333,但是因为该属性是短暂的,该值没有被发送到输出流。所以反序列化后 Employee 对象的 SSN 属性为 0。

AI 思考中...

5 篇笔记 写笔记

  1. #0

    小怪物

    193***[email protected]

    35

    Serializable 的作用

    为什么一个类实现了Serializable接口,它就可以被序列化呢?在上节的示例中,使用ObjectOutputStream来持久化对象,在该类中有如下代码:

    private void writeObject0(Object obj, boolean unshared) throws IOException {
     ...
     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());
     }
     }
     ...
    } 

    从上述代码可知,如果被写对象的类型是String,或数组,或Enum,或Serializable,那么就可以对该对象进行序列化,否则将抛出NotSerializableException。

    小怪物

    193***[email protected]

    9年前 (2018年02月08日)
  2. #0

    hunter

    hun***[email protected]

    36

    关于 java 中的序列化与反序列化

    关于序列化,常又称为持久化,将其写入磁盘中。

    进而对于编码规则来说:

    任一一个实体类必须要去实现 Serializable 接口,方便以后将该类持久化,或者将其用于转为字节数组,用于网络传输。

    对于一个实体类,不想将所有的属性都进行序列化,有专门的关键字 transient:

    private transient String name;

    当对该类序列化时,会自动忽略被 transient 修饰的属性。

    hunter

    hun***[email protected]

    8年前 (2018年03月25日)
  3. #0

    关于 SerializableID

    SerializableID 号是根据类的特征和类的签名算出来的。为什么 ID 号那么长,是因为为了避免重复。所以 Serializable 是给类加上 id 用的。用于判断类和对象是否是同一个版本。

    如果可序列化类未显式声明 serialVersionUID,则序列化运行时将基于该类的各个方面计算该类的默认 serialVersionUID 值。原因是计算默认的 serialVersionUID 对类的详细信息具有较高的敏感性,根据编译器实现的不同可能千差万别,这样在反序列化过程中可能会导致意外的 InvalidClassException。

    8年前 (2019年01月12日)
  4. #0

    ByVie

    635***[email protected]

    70

    序列化流与反序列化流

    ObjectOutputStream(序列化流)

    ObjectOutputStream是序列化流,可以将Java程序中的对象写到文件中。

    ObjectOutputStream 构造方法:

    ObjectOutputStream(OutputStream out):参数要传递字节输出流。

    ObjectOutputStream写对象的方法(特有方法):

    void writeObject(Object obj): 向文件中写对象。

    ObjectOutputStream 的使用步骤:

    • 创建序列化流,用来写。
    • 调用 writeObject 方法,写对象。
    • 释放资源。

    tips: 要使用序列化流向文件中写的对象,必须实现 Serializable 接口。

    public class Demo01ObjectOutputStream {
     public static void main(String[] args) throws IOException {
     //1. 创建序列化流,用来写
     ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("day12\\file03-obj.txt"));
     //2. 调用writeObject方法,写对象
     Person p = new Person("张三丰", 100);
     oos.writeObject(p);
     //3. 释放资源。
     oos.close();
     }
    }

    ObjectInputStream(反序列化流)

    ObjectInputStream 是反序列化流, 可以将文件中的对象读取到 Java 程序中。

    ObjectInputStream 的构造方法:

    ObjectInputStream(InputStream in):参数要传递字节输入流。

    ObjectInputStream 读取对象的方法(特有的方法):

    Object readObject(): 从文件中读取对象,并将该对象返回。

    反序列化流的使用步骤:

    • 创建 ObjectInputStream 反序列化流。
    • 调用 readObject 方法,读取对象。
    • 释放资源。

    tips:调用 readObject 方法读取对象时,对象所对应的类不存在,那么会报错(ClassNotFoundException)

    特殊情况:

    被 static 修饰的成员变量无法序列化,无法写到文件。

    如果不希望某个成员变量写到文件,同时又不希望使用 static 关键字, 那么可以使用 transient。transient 关键字表示瞬态,被 transient 修饰的成员变量无法被序列化。

    public class Demo03StaticAndTransient {
     public static void main(String[] args) throws IOException, ClassNotFoundException {
     writePerson();
     readPerson();
     }
     //从文件中读取Person对象
     public static void readPerson() throws IOException, ClassNotFoundException {
     //创建反序列化流
     ObjectInputStream ois = new ObjectInputStream(new FileInputStream("day12\\file04-obj.txt"));
     //从文件中读取对象
     Object obj = ois.readObject();
     System.out.println(obj);
     //释放资源
     ois.close();
     }
     //向文件中写Person对象
     public static void writePerson() throws IOException {
     //创建序列化流
     ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("day12\\file04-obj.txt"));
     //向文件中写Person对象
     oos.writeObject(new Person("张三丰", 100));
     //关流
     oos.close();
     }
    }
    public class Demo04SerialVersionUID {
     public static void main(String[] args) throws IOException, ClassNotFoundException {
     //writePerson();
     readPerson();
     }
     //从文件中读取Person对象
     public static void readPerson() throws IOException, ClassNotFoundException {
     //创建反序列化流
     ObjectInputStream ois = new ObjectInputStream(new FileInputStream("day12\\file04-obj.txt"));
     //从文件中读取对象
     Object obj = ois.readObject();
     System.out.println(obj);
     //释放资源
     ois.close();
     }
     //向文件中写Person对象
     public static void writePerson() throws IOException {
     //创建序列化流
     ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("day12\\file04-obj.txt"));
     //向文件中写Person对象
     oos.writeObject(new Person("张三丰", 100));
     //关流
     oos.close();
     }
    }

    例题

    /*
     练习:
     1. 将存有多个Student对象的集合序列化操作,保存到list.txt 文件中。
     2. 反序列化list.txt ,并遍历集合,打印对象信息。
     步骤:
     1. 创建集合,用来保存Student
     2. 向集合中添加Student对象。
     3. 创建ObjectOutputStream序列化流,用来写。
     4. 调用writeObject方法,向文件中写集合对象。
     5. 释放资源。
     6. 创建ObjectInputStream反序列化流对象,用来读取
     7. 调用readObject方法,从文件中读取对象。
     8. 将读取到的集合进行遍历,并输出结果。
     注意:如果想要将多个对象保存在文件中,最好的一个方式可以将多个对象放入到一个集合中,然后直接将集合写到文件中。
     */
    public class Demo05Test {
     public static void main(String[] args) throws IOException, ClassNotFoundException {
     //1. 创建集合,用来保存Student
     List<Student> list = new ArrayList<>();
     //2. 向集合中添加Student对象。
     list.add(new Student("李云龙", 20));
     list.add(new Student("二营长", 22));
     list.add(new Student("秀琴", 25));
     //3. 创建ObjectOutputStream序列化流,用来写。
     ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("day12\\list.txt"));
     //4. 调用writeObject方法,向文件中写集合对象。
     oos.writeObject(list);
     //5. 释放资源。
     oos.close();
     //6. 创建ObjectInputStream反序列化流对象,用来读取
     ObjectInputStream ois = new ObjectInputStream(new FileInputStream("day12\\list.txt"));
     //7. 调用readObject方法,从文件中读取对象。
     List<Student> list2 = (List<Student>) ois.readObject();
     //8. 将读取到的集合进行遍历,并输出结果。
     for (Student stu : list2) {
     System.out.println(stu);
     }
     }
    }
    ByVie

    ByVie

    635***[email protected]

    7年前 (2019年08月16日)
  5. #0

    我是渣渣

    156***[email protected]

    参考地址

    15

    1)一旦变量被transient修饰,变量将不再是对象持久化的一部分,该变量内容在序列化后无法获得访问。

    2)transient关键字只能修饰变量,而不能修饰方法和类。注意,本地变量是不能被transient关键字修饰的。变量如果是用户自定义类变量,则该类需要实现Serializable接口。

    3)被transient关键字修饰的变量不再能被序列化,一个静态变量不管是否被transient修饰,均不能被序列化。

    我是渣渣

    156***[email protected]

    参考地址

    6年前 (2020年12月04日)

点我分享笔记

  • 昵称 (必填)
  • 邮箱 (必填)
  • 引用地址

AltStyle によって変換されたページ (->オリジナル) /