为什么使用Serializable(序列化)

  经常看到有些类调用了Serializable接口,而有些类又没有调用Serializable接口。那么什么情况下要调用Serializable接口。

    首先来了解一下Serializable。(类通过实现 java.io.Serializable 接口以启用其序列化功能。未实现此接口的类将无法使其任何状态序列化或反序列化。序列化接口Serializable没有方法或字段,仅用于标识可序列化的语义)


    实现了Serializable接口的对象,可将它们转换成一系列字节,并可在以后完全恢复回原来的样子。这一过程亦可通过网络进行。这意味着序列化机制能自动补偿操作系统间的差异。换句话说,可以先在Windows机器上创建一个对象,对其序列化,然后通过网络发给一台Unix机器,然后在那里准确无误地重新“装配”。不必关心数据在不同机器上如何表示,也不必关心字节的顺序或者其他任何细节。


    serialization主要用来支持2种主要的特性:
    1、RMI(Remote method invocation)。RMI允许象在本机上一样操作远程机器上的对象。当发送消息给远程对象和调用远程方法时,就需要用到serializaiton机制来发送参数和接收返回值。
    2、保存信息。在某个时候把状态信息保存起来,以便以后某个时候能恢复这些状态信息。
    Hibernaet和EJB中的实体Bean就用到了上面两个特性。


    另外:保存的时候不仅能保存对象的副本,而且还会把对象里面所引用的对象也保存起来,以此类推。就像在编译某个类一样,会涉及到所用到的所有类。但是所引用的对象也必须是可序列化的,不然会抛NotSerializableException异常。

    
    下面来写个例子:(A和B类都是可序列化的,WriteObj:将A序列化,ReadObj:将A反序列化) 


package woxingwosu;

import java.io.Serializable;

public class A implements Serializable{
    
private String name="my name is a";
    
private B b=null;
    A(){
        b
=new B();
    }
    
public B getB() {
        
return b;
    }
    
public void setB(B b) {
        
this.b = b;
    }
    
public String getName() {
        
return name;
    }
    
public void setName(String name) {
        
this.name = name;
    }
    
public String show(){
        
return "a.toString <a.name=/""+this.name+"/" a.b.name=/""+this.b.getName()+"/">"
            
+"/na="+this.toString()+" b="+this.b.toString();
    }
}

 

class B

 

class WriteSeri

 

class ReadSeri

    首先,我们运行WriteObj,实现序列化,得到输出结果

write obj :a.toString  < a.name = " my name is a "  a.b.name = " my name is B " >
a
= woxingwosu.A@a90653 b = woxingwosu.B@de6ced

    然后我们再运行ReadObj,实现反序列化,得到输出结果

read Object :a.toString  < a.name = " my name is a "  a.b.name = " my name is B " >
a
= woxingwosu.A@a90653 b = woxingwosu.B@de6ced



    遗漏了一个问题,就是标识版本的serialVersionUID。
    官方文档:如果可序列化类未显式声明 serialVersionUID,则序列化运行时将基于该类的各个方面计算该类的默认 serialVersionUID 值,如“Java(TM) 对象序列化规范”中所述。不过,强烈建议 所有可序列化类都显式声明 serialVersionUID 值,原因计算默认的 serialVersionUID 对类的详细信息具有较高的敏感性,根据编译器实现的不同可能千差万别,这样在反序列化过程中可能会导致意外的 InvalidClassException。因此,为保证 serialVersionUID 值跨不同 java 编译器实现的一致性,序列化类必须声明一个明确的 serialVersionUID 值。还强烈建议使用 private 修改器显示声明 serialVersionUID(如果可能),原因是这种声明仅应用于立即声明类 -- serialVersionUID 字段作为继承成员没有用处。

    刚才写的例子中就没有用到serialVersionUID,这时JVM会根据类名、接口名、成员方法及属性等来生成一个64位的哈希字段作为serialVersionUID。但是如果序列化和反序列化的JVM版本不一样的话,还是显示写上serialVersionUID安全。

Serializable 序列化是将一个对象转换为字节序列的过程,可以用于对象的持久化、网络传输等场景。以下是一个使用 Serializable 序列化的案例: 假设我们有一个 Person 类,包含姓名和年龄两个属性: ```java public class Person implements Serializable { private String name; private int age; public Person(String name, int age) { this.name = name; this.age = age; } // getter 和 setter 方法省略 } ``` 现在我们想将一个 Person 对象保存到文件中,可以使用 ObjectOutputStream 进行序列化: ```java Person person = new Person("Tom", 18); try (ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("person.dat"))) { oos.writeObject(person); // 将 Person 对象写入文件 } catch (IOException e) { e.printStackTrace(); } ``` 注:try-with-resources 语句会在代码执行完成后自动关闭流,无需手动调用 close 方法。 接下来,我们可以从文件中读取 Person 对象并进行反序列化: ```java try (ObjectInputStream ois = new ObjectInputStream(new FileInputStream("person.dat"))) { Person person = (Person) ois.readObject(); // 从文件中读取 Person 对象 System.out.println(person.getName() + " " + person.getAge()); // 输出:Tom 18 } catch (IOException | ClassNotFoundException e) { e.printStackTrace(); } ``` 注:ObjectInputStream.readObject 方法会抛出 ClassNotFoundException 异常,需要进行捕获或声明抛出。 通过 Serializable 序列化,我们可以方便地将对象保存到文件中并进行读取,从而实现对象的持久化,这在很多应用场景下非常有用。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值