序列化_01

学习目标:

提示:这里可以添加学习目标

例如:

  • 一周掌握 Java 入门知识

学习内容:

3.1 概述

java提供了一种对象序列化的机制,用一个字节序列可以表示一个对象,该字节序列包含对象的数据、对象的类型和对象的属性等信息。字节序列写出到文件之后,相当于文件中持久保持了一个对象的信息。
反之,该字节序列还可以从文件中读取回来,重构对象,对它进行反序列化。对象的数据、对象的类型和对象中存储的数据信息,都可以用来在内存中创建对象。
对象的序列化:把对象以流的的方式,写入到文件中保存,叫写对象也叫对象的序列化
对象的反序列化:把文件中保存的对象,以流的方法读取出来,叫做读对象,也叫对象的反序列化

3.2 对象序列化流 ObjectOutputStream

java.io.ObjectOutputStream extends OutputStream
ObjectOutputStream:对象的序列化流
作用:把对象以流的方式写入到文件中保存
构造方法:
ObjectOutputStream(OutputStream out) 创建写入指定 OutputStream的ObjectoutputStream.
参数:
OutputStream out :字节输出流
特有的成员方法:
void writeObject(Object obj) 将指定的对象写入 ObjectOutputStream.
使用步骤:

  1. 创建ObjectOutputStream对象,构造方法中传递字节输出流
  2. 使用ObjectOutputStream对象中的方法writeObject,把对象写入到文件中
  3. 释放资源
    注意:序列化和反序列化的时候,会抛出NotSerializableException没有序列化异常
    类通过实现java.io.Serializable接口,就会给类添加一个标记
    当我们进行序列化和反序列化的时候,就会检测类上是否有这个标记
    有:就可以序列化
    没有:就会抛出NotSerializableException异常

static关键字:静态关键字
静态优先与非静态加载到内存中(静态优先于对象进入到内存中)
被static修饰的成员变量不能被序列化,序列化的都是对象
private static int age;
oos.writeObject(new Person(“A”,18))
Object o=ois.readObject();
Person(name =“A”,age =0);
transient关键字:瞬态关键字
被transient修饰成员变量,不能被序列化
private transient int age;
oos.writeObject(new Person(“A”,18));
Object o =ois.readObject();
Person(name =“A”,age =0)
代码案例
1.对象类

package day22.Text10;

import java.io.Serializable;

public class Person implements Serializable {
    private int age;
    private String name;

    public Person(int age, String name) {
        this.age = age;
        this.name = name;
    }

    @Override
    public String toString() {
        return "Person{" +
                "age=" + age +
                ", name='" + name + '\'' +
                '}';
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public Person() {
    }
}

主类

package day22.Text10;

import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.ObjectOutputStream;

public class Demo_01 {
    public static void main(String[] args) throws IOException {
        ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("D:\\IdeaProjects\\untitled\\a.txt"));
        oos.writeObject(new Person(18,"A"));
        oos.close();
    }
}

3.3 对象的反序列化流

java.io.ObjectInputStream extends InputStream
ObjectInputStream:对象的反序列化流
作用:把文件中保存的对象,以流的方式读取出来使用

构造方法:
ObjectInputStream(InputStream in) 创建指定 InputStream 读取的ObjectInputStream.
参数:
InputStream in:字节输入流
特有的成员方法:
Object readObject( ) 从ObjectInputStream读取对象。
使用步骤:

  1. 创建ObjectInpurStream对象,构造方法中传递字节输入流
  2. 使用ObjectInputStream对象中的方法readObject读取保存对象的文件
  3. 释放资源
  4. 使用读取出来的对象(打印)
    注意事项:
    readObject方法声明抛出了ClassNotFoundException(class文件找不到异常)
    当不存在对象的class文件时抛出此异常
    反序列化的前提:
    类必实现Serializanle
    必须存在对应的class文件

反序列化操作2

  • 当JVM反序列化对象时,能找到class文件,但是class文件在序列化对象之后发生了修改,那么反序列化操作也会失败,抛出一个InvalidClassException异常。发生这个异常的原因如下:
    1、该类的序列版本号与从流中读取的类描述的版本号不匹配
    2、该类包含未知的数据类型
    3、该类没有可访问的无参数构造方法
    Serializable 接口需要序列化的类,提供了一个序列版本号。seriaVersionUID该版本号的目的在于验证序列化的对象和对应类是否版本匹配
    问题:
    每次修改类的定义,都会给class文件生成一个序列号
    解决方案:
    无论是否对类的定义进行修改,都不重新生成新的序列号
    可以手动给类添加一个序列号
    格式在Serializable接口规定:
    可序列化类可以通过声明为"serialVersionUID"的字段(该字段必须是静态(static)、最终(final)的long型字段)显示声明其自己的serialVersionUID:
    eg: static final long serialVersionUID =43L(常量 不能改变)

代码案例

package day22;

import day22.Text10.Person;

import java.io.FileInputStream;
import java.io.IOException;
import java.io.ObjectInputStream;

public class Text11 {
    public static void main(String[] args) throws IOException, ClassNotFoundException {
//        1. 创建ObjectInpurStream对象,构造方法中传递字节输入流
        ObjectInputStream ois = new ObjectInputStream(new FileInputStream("D:\\IdeaProjects\\untitled\\a.txt"));
//        2. 使用ObjectInputStream对象中的方法readObject读取保存对象的文件
        Object o = ois.readObject();
//        3. 释放资源
        ois.close();
//        4. 使用读取出来的对象(打印)
        System.out.println(o);
    }
}

综合练习:序列化集合

当我们想在文件中保存多个对象的时候
可以把多个对象存储到一个集合中
对集合进行序列化和反序列化
分析:
1、定义一个存储Person对象的ArrayList集合
2、往ArrayList集合中存储Person对象
3、创建一个序列化流ObjectOutputStream对象
4、使用ObjectOutputStream对象中的方法writeObject,对集合进行序列化
5、创建一个反序列化ObjectInputStream对象
6、使用ObjectInputStream对象中的方法readObject读取文件中保存的集合
7、把Object类型的集合转换为ArrayList类型
8、遍历ArrayList集合
9、释放资源

package day22;

import day22.Text10.Person;

import java.io.*;
import java.util.ArrayList;

public class Text12 {
    public static void main(String[] args) throws IOException, ClassNotFoundException {
//        1、定义一个存储Person对象的ArrayList集合
        ArrayList<Person> list = new ArrayList<>();
//        2、往ArrayList集合中存储Person对象
        list.add(new Person(19,"A"));
        list.add(new Person(20,"B"));
        list.add(new Person(21,"C"));
//        3、创建一个序列化流ObjectOutputStream对象
        list.add(new Person(22,"D"));
        ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("D:\\IdeaProjects\\untitled\\a.txt"));
//        4、使用ObjectOutputStream对象中的方法writeObject,对集合进行序列化
        oos.writeObject(list);
//        5、创建一个反序列化ObjectInputStream对象
        ObjectInputStream ois = new ObjectInputStream(new FileInputStream("D:\\IdeaProjects\\untitled\\a.txt"));
//        6、使用ObjectInputStream对象中的方法readObject读取文件中保存的集合
        Object o = ois.readObject();
//        7、把Object类型的集合转换为ArrayList类型
        ArrayList<Person> list1=(ArrayList<Person>)o;
//        8、遍历ArrayList集合
        for (Person p:list1) {
            System.out.println(p);
        }
//        9、释放资源
        ois.close();
        oos.close();
    }
}


学习时间:

提示:这里可以添加计划学习的时间

例如:

  • 周一至周五晚上 7 点—晚上9点
  • 周六上午 9 点-上午 11 点
  • 周日下午 3 点-下午 6 点

学习产出:

提示:这里统计学习计划的总量

例如:

  • 技术笔记 2 遍
  • CSDN 技术博客 3 篇
  • 习的 vlog 视频 1 个
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值