Java---transient关键字

序列化 (Serialization): 是将对象的状态信息转换为可以存储或传输的形式的过程。在序列化期间,对象将其当前状态写入到临时或持久性存储区。以后,可以通过从存储区中读取或反序列化对象的状态,重新创建该对象。

反序列化: 反序列化从序列化的表示形式中提取数据,并直接设置对象状态,这也与可访问性规则无关。

其实通俗一点的解释,序列化就是把一个对象保存到一个文件或数据库字段中去,反序列化就是在适当的时候把这个文件再转化成原来的对象使用。

transient关键字的作用是:被transient修饰的变量不参与序列化和反序列化。当一个对象被序列化的时候,transient型变量的值不包括在序列化的表示中,然而非transient型的变量是被包括进去的。

Java的serialization提供了一种持久化对象实例的机制。当持久化对象时,可能有一个特殊的对象数据成员,我们不想用serialization机制来保存它。

为了在一个特定对象的一个域上关闭serialization,可以在这个域前加上关键字transient。

当一个对象被序列化的时候,transient型变量的值不包括在序列化的表示中,然而非transient型的变量是被包括进去的。

简而言之,被transient修饰的变量不参与序列化和反序列化。

接下来用代码来证明一下。

新建一个Student类实现Serializable 接口,并重写其toString方法便于观察结果。

一个age属性不被transient修饰,一个name属性被transient修饰。

public class Student implements Serializable {
 
    private int age;
 
    private transient String name;
 
 
    public Student() {
 
    }
 
 
    public Student(int age, String name) {
 
        this.age = age;
 
        this.name = name;
 
    }
 
 
    @Override
 
    public String toString() {
 
        return "Student{" + "age=" + age + ", name='" + name + '\'' + '}';
    }
}

然后在TransientTest类里边测试。

为了代码简洁这里的IO操作没有进行try catch操作而是直接抛出了。

public class TestTransient {
 
 
    public static void main(String[] args) throws Exception {
 
        // 实例化一个Student对象.
 
        Student student = new Student(15, "HuaGe");
 
        System.out.println(student);
 
 
        // 将student对象写入磁盘文件(序列化)
 
        ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("student.txt"));
 
        oos.writeObject(student);
 
        oos.close();
 
 
        // 从磁盘文件读取student对象(反序列化)
 
        ObjectInputStream ois = new ObjectInputStream(new FileInputStream("student.txt"));
 
        student = (Student) ois.readObject();
 
        System.out.println(student);
 
    }
 
}

运行main方法,观察控制台打印信息
请添加图片描述

发现经过了序列化和反序列化后,name属性从HuaGe变为了null.

这就说明了被transient修饰的变量不参与序列化和反序列化。那有没有例外呐?

我们知道,java中有两种序列化的方式。

  1. 实现Serializable接口。

  2. 实现Externalizable接口。

Externalizable接口是Serializable接口的子类

public interface Externalizable extends java.io.Serializable {
 
    void writeExternal(ObjectOutput out) throws IOException;
 
    void readExternal(ObjectInput in) throws IOException, ClassNotFoundException;
 
}

这个接口的两个方法可以指定对类中的哪些属性进行序列化。

使用这个接口时,无论属性有没有被transient修饰,

默认不对任何属性进行序列化。所以实现了Externalizable接口的类

一般不再使用transient修饰属性。

总结:

  1. 被transient修饰的变量不参与序列化和反序列化

  2. transient一般在实现了Serializable接口的类中使用。

原文地址: https://www.php.cn/java/base/467138.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值