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

  • 7
    点赞
  • 58
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值