Java-transient瞬态关键字

简介

当一个类的对象需要被序列化时,但是某些属性不需要被序列化,这时不需要序列化的属性可以使用关键字transient 修饰。
只要被关键字transient 修饰了,序列化时这个属性就不会被序列化了。同时静态修饰也不会被序列化,因为序列化是把对象数据进行持久化存储,而静态的属于类加载时的数据,不会被序列化。


代码

Person 类

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
package cn.vgbhfive;

import java.io.Serializable;

/**
* @auther Vgbh
*
*/

public class Person implements Serializable{

/**
*
*/
private static final long serialVersionUID = 1L;

private Integer id;

private transient String name;

private String firstName;

public Person(Integer id, String name, String firstName) {
this.id = id;
this.name = name;
this.firstName = firstName;
}

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

public static void main(String[] args) {
Person p1 = new Person(12, "ssss", "ddd");
SerializationUtils.writeObject(p1);

Person p2 = (Person) SerializationUtils.readObject();
System.out.println(p2.toString());
}


}

SerializationUtils 类

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
package cn.vgbhfive;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.Serializable;

/**
* @auther Vgbh
*
*/

public class SerializationUtils {

private static final String FILENAME = "F:/obj.bin";

/**
* 序列化
* @param s
*/
public static void writeObject(Serializable s) {
try {
ObjectOutputStream out = new ObjectOutputStream(new FileOutputStream(FILENAME));
out.writeObject(s);
out.close();
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}

/**
* 反序列化
* @return
*/
public static Object readObject() {
Object obj = null;
try {
ObjectInputStream in = new ObjectInputStream(new FileInputStream(FILENAME));
try {
obj = in.readObject();
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
in.close();
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
return obj;
}
}

执行结果


从执行结果就可以看出,瞬态变量没有被序列化,因此在反序列化时,name 的值为null 。


注意

下面这些是在实际编码过程中,可能会遇到的问题的解决方法,希望会对你有用。

  • 显式声明UID
    在序列化和反序列化的类不一致的情形下,反序列化回报一个InvalidClassException 异常,原因就是序列化和反序列化所对应的类版本发生了变化。
  • 避免用序列化类在构造函数中为不变量赋值
    反序列化时构造函数不会执行。
  • 避免为final变量复杂赋值
    这里的赋值指的是“简单对象”,简单对象包括:8个基本类型、数组、字符串。
  • 使用序列化类的私有方法巧妙解决部分属性持久化问题
    实现了Serialization 接口的类可以实现两个私有方法:writeObject() 和readObject() ,以影响和控制序列化和反序列化的过程。
  • 反序列化时以下情况不会被重新赋值:通过构造函数为final 变量赋值、通过方法返回值为final 变量赋值、final 修饰的属性不是基本类型。