IO流(八) 序列化流与反序列化流

用于从流中读取对象的操作流 ObjectInputStream    称为 反序列化流

用于向流中写入对象的操作流 ObjectOutputStream   称为 序列化流

序列化:用一个字节序列可以表示一个对象,该字节序列包含该对象的数据、对象的类型和对象中存储的属性等信息。字节序列写出到文件之后,相当于文件中持久保存了一个对象的信息。

反序列化:该字节序列还可以从文件中读取回来,重构对象,对它进行反序列化。对象的数据、对象的类型和对象中存储的数据信息,都可以用来在内存中创建对象。

特点:用于操作对象。可以将对象写入到文件中,也可以从文件中读取对象。

对象序列化流ObjectOutputStream

ObjectOutputStream 将 Java 对象的基本数据类型和图形写入 OutputStream。可以使用 ObjectInputStream 读取(重构)对象。通过在流中使用文件可以实现对象的持久存储。

注意:只能将支持 java.io.Serializable 接口的对象写入流中

java.io.ObjectOutputStream extends OutputStream

ObjectOutputStream:对象的序列化流

作用:把对象以流的方式写入到文件中保存

 

构造方法:

    ObjectOutputStream(OutputStream out) 创建写入指定 OutputStream 的 ObjectOutputStream。

    参数:

        OutputStream out:字节输出流

特有的成员方法:

    void writeObject(Object obj) 将指定的对象写入 ObjectOutputStream。

 

使用步骤:

    1.创建ObjectOutputStream对象,构造方法中传递字节输出流

    2.使用ObjectOutputStream对象中的方法writeObject,把对象写入到文件中

    3.释放资源

代码演示:

public class ObjectStreamDemo {
	public static void main(String[] args) throws IOException, ClassNotFoundException {
		/*
		 * 将一个对象存储到持久化(硬盘)的设备上。
		 */
		writeObj();//对象的序列化。
	}
	public static void writeObj() throws IOException {
		//1,明确存储对象的文件。
		FileOutputStream fos = new FileOutputStream("tempfile\\obj.object");
		//2,给操作文件对象加入写入对象功能。
		ObjectOutputStream oos = new ObjectOutputStream(fos);
		//3,调用了写入对象的方法。
		oos.writeObject(new Person("wangcai",20));
		//关闭资源。
		oos.close();
	}
}

Person类

public class Person implements Serializable {
	private String name;
	private int age;
	public Person() {
		super();
	}
	public Person(String name, int age) {
		super();
		this.name = name;
		this.age = age;
	}

	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
	public int getAge() {
		return age;
	}
	public void setAge(int age) {
		this.age = age;
	}
	@Override
	public String toString() {
		return "Person [name=" + name + ", age=" + age + "]";
	}
}

对象反序列化流ObjectInputStream

ObjectInputStream 对以前使用 ObjectOutputStream 写入的基本数据和对象进行反序列化。支持 java.io.Serializable接口的对象才能从流读取。

java.io.ObjectInputStream extends InputStream

ObjectInputStream:对象的反序列化流

作用:把文件中保存的对象,以流的方式读取出来使用

 

构造方法:

    ObjectInputStream(InputStream in) 创建从指定 InputStream 读取的 ObjectInputStream。

    参数:

        InputStream in:字节输入流

特有的成员方法:

    Object readObject() 从 ObjectInputStream 读取对象。

 

使用步骤:

    1.创建ObjectInputStream对象,构造方法中传递字节输入流

    2.使用ObjectInputStream对象中的方法readObject读取保存对象的文件

    3.释放资源

    4.使用读取出来的对象(打印)

 

 readObject方法声明抛出了ClassNotFoundException(class文件找不到异常)

 当不存在对象的class文件时抛出此异常

 反序列化的前提:

    1.类必须实现Serializable

    2.必须存在类对应的class文件

代码演示:

public class ObjectStreamDemo {
	public static void main(String[] args) throws IOException, ClassNotFoundException {
		readObj();//对象的反序列化。
	}
	public static void readObj() throws IOException, ClassNotFoundException {
		
		//1,定义流对象关联存储了对象文件。
		FileInputStream fis = new FileInputStream("tempfile\\obj.object");
		
		//2,建立用于读取对象的功能对象。
		ObjectInputStream ois = new ObjectInputStream(fis);
		
		Person obj = (Person)ois.readObject();
		
		System.out.println(obj.toString());
		
	}
}

序列化接口

当一个对象要能被序列化,这个对象所属的类必须实现Serializable接口。否则会发生异常NotSerializableException异常。

同时当反序列化对象时,如果对象所属的class文件在序列化之后进行的修改,那么进行反序列化也会发生异常InvalidClassException。发生这个异常的原因如下:

  1. 该类的序列版本号与从流中读取的类描述符的版本号不匹配
  2. 该类包含未知数据类型
  3. 该类没有可访问的无参数构造方法

Serializable标记接口。该接口给需要序列化的类,提供了一个序列版本号。serialVersionUID. 该版本号的目的在于验证序列化的对象和对应类是否版本匹配。

 

代码修改如下,修改后再次写入对象,读取对象测试

public class Person implements Serializable {
	//给类显示声明一个序列版本号。
	private static final long serialVersionUID = 1L;
	private String name;
	private int age;
	public Person() {
		super();
		
	}
	public Person(String name, int age) {
		super();
		this.name = name;
		this.age = age;
	}

	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
	public int getAge() {
		return age;
	}
	public void setAge(int age) {
		this.age = age;
	}
	@Override
	public String toString() {
		return "Person [name=" + name + ", age=" + age + "]";
	}
}

瞬态关键字transient

当一个类的对象需要被序列化时,某些属性不需要被序列化,这时不需要序列化的属性可以使用关键字transient修饰。只要被transient修饰了,序列化时这个属性就不会琲序列化了。

同时静态修饰也不会被序列化,因为序列化是把对象数据进行持久化存储,而静态的属于类加载时的数据,不会被序列化。

static关键字:静态关键字

    静态优先于非静态加载到内存中(静态优先于对象进入到内存中)

    被static修饰的成员变量不能被序列化的,序列化的都是对象

    private static int age;

    oos.writeObject(new Person("小美女",18));

    Object o = ois.readObject();

    Person{name='小美女', age=0}

 

transient关键字:瞬态关键字

    被transient修饰成员变量,不能被序列化

    private transient int age;

    oos.writeObject(new Person("小美女",18));

    Object o = ois.readObject();

    Person{name='小美女', age=0}

代码修改如下,修改后再次写入对象,读取对象测试

public class Person implements Serializable {
	/*
	 * 给类显示声明一个序列版本号。
	 */
	private static final long serialVersionUID = 1L;
	private static String name;
	private transient/*瞬态*/ int age;
	
	public Person() {
		super();
		
	}
	
	public Person(String name, int age) {
		super();
		this.name = name;
		this.age = age;
	}

	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
	public int getAge() {
		return age;
	}
	public void setAge(int age) {
		this.age = age;
	}

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

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值