在实际开发当中我们经常需要对一些对象进行序列化操作,但是有时我们一个对象中有些敏感的字段、或者业务要求默写数据不能进行序列化存储,那我们怎么办呢?这时候transient就可以帮我们解决类似的问题了。
transient的使用方法也是很简单,我们只需要在不需要序列化的字段声明时加上transient关键字进行修饰即可,此时被transient关键字修饰的字段的值就不会被序列化了。
让我们来通过实际的代码来体会一下transient的功能吧!
首先我们创建一个用于测试的User类:
package com.mark.java;
import java.io.Serializable;
/**
* Created by Mark on 16/8/11.
*/
public class User implements Serializable {
private String name;
private String password;
public User(String name, String password) {
this.name = name;
this.password = password;
}
public String getName() {
return name;
}
public User setName(String name) {
this.name = name;
return this;
}
public String getPassword() {
return password;
}
public User setPassword(String password) {
this.password = password;
return this;
}
@Override
public String toString() {
return "User{" +
"name='" + name + '\'' +
", password='" + password + '\'' +
'}';
}
}
User类实现了Serializable接口,表示可以进行序列化操作,我们来看看不使用transient关键字进行序列化和反序列化的效果:
package com.mark.java;
import android.os.Bundle;
import android.os.Environment;
import android.support.v7.app.AppCompatActivity;
import android.util.Log;
import android.view.View;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
public class MainActivity extends AppCompatActivity {
private static final String TAG = MainActivity.class.getSimpleName();
/**
* 存储User对象的路径
*/
private String path = Environment.getExternalStorageDirectory().getAbsolutePath() + File.separator + "user.text";
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
// 序列化操作
findViewById(R.id.btnWrite).setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
// 创建User对象
User user = new User("Mark", "coder");
// 打印User对象
Log.d(TAG, "Write : " + user.toString());
// 将User对象写入到指定的文件中
try {
File file = new File(path);
if (!file.exists()) {
file.createNewFile();
}
ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream(file));
oos.writeObject(user);
oos.flush();
oos.close();
} catch (IOException e) {
e.printStackTrace();
}
}
});
// 反序列化操作
findViewById(R.id.btnRead).setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
// 读取上一步写入文件中的User对象
try {
ObjectInputStream ois = new ObjectInputStream(new FileInputStream(path));
User user = (User) ois.readObject();
ois.close();
// 打印获取到的User对象
Log.d(TAG, "Read : " + user.toString());
} catch (IOException e) {
e.printStackTrace();
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
}
});
}
}
上面的代码根据注释大家都能看懂,这里就不解释了,看看操作后的日志输出情况吧。
08-11 02:21:46.124 10302-10302/com.mark.java D/MainActivity: Write : User{name='Mark', password='coder'}
08-11 02:21:48.839 10302-10302/com.mark.java D/MainActivity: Read : User{name='Mark', password='coder'}
可以看到,序列化和反序列化操作之后User的信息是相同的,假设这时我们不想讲password字段进行序列化,那我们就需要使用transient关键字进行修饰,例如:
private transient String password;
我们在重复上面的操作,日志如下:
08-11 02:26:54.088 15038-15038/com.mark.java D/MainActivity: Write : User{name='Mark', password='coder'}
08-11 02:26:55.855 15038-15038/com.mark.java D/MainActivity: Read : User{name='Mark', password='null'}
可以看到password=null了,这就是transient的作用,因为上面我们使用transient关键字修饰了password字段,所以在序列化操作的时候系统是不会对password进行序列化操作的,而是给他一个null。
我们再来测试一下基本数据类型被transient修饰会怎样。
package com.mark.java;
import java.io.Serializable;
/**
* Created by Mark on 16/8/11.
*/
public class User implements Serializable {
private transient boolean bool = false;
private transient long l = 1234567890L;
private transient double d = 123456789.0D;
private transient float f = 123456789.0F;
private transient short s = 123;
private transient byte b = 1;
private transient int age = 20;
private String name;
private transient String password;
public User(String name, String password) {
this.name = name;
this.password = password;
}
public String getName() {
return name;
}
public User setName(String name) {
this.name = name;
return this;
}
public String getPassword() {
return password;
}
public User setPassword(String password) {
this.password = password;
return this;
}
@Override
public String toString() {
return "User{" +
"bool=" + bool +
", l=" + l +
", d=" + d +
", f=" + f +
", s=" + s +
", b=" + b +
", age=" + age +
", name='" + name + '\'' +
", password='" + password + '\'' +
'}';
}
}
我们在User中简单的增加了一些基本数据类型的测试数据,并重写了toString方法。打印日志如下:
08-11 02:37:22.256 24595-24595/com.mark.java D/MainActivity: Write : User{bool=false, l=1234567890, d=1.23456789E8, f=1.2345679E8, b=1, age=20, name='Mark', password='coder'}
08-11 02:37:24.671 24595-24595/com.mark.java D/MainActivity: Read : User{bool=false, l=0, d=0.0, f=0.0, b=0, age=0, name='Mark', password='null'}
根据日志可以得出结论,所有被transient修饰的字段,不会被序列化,在反序列化时此字段会使用此字段类型的默认值。
修改User类如下:
package com.mark.java;
import java.io.Serializable;
/**
* Created by Mark on 16/8/11.
*/
public class User implements Serializable {
private final boolean bool = true;
private final long l = 1234567890L;
private static double d = 123456789.0D;
private static float f = 123456789.0F;
private short s = 123;
private byte b = 1;
private int age = 20;
private static String name;
private final String name1 = "Mark";
private static final String name2 = "Mark";
private String password;
public User(String name, String password) {
this.name = name;
this.password = password;
}
public String getName() {
return name;
}
public User setName(String name) {
this.name = name;
return this;
}
public String getPassword() {
return password;
}
public User setPassword(String password) {
this.password = password;
return this;
}
@Override
public String toString() {
return "User{" +
"bool=" + bool +
", l=" + l +
", d=" + d +
", f=" + f +
", s=" + s +
", b=" + b +
", age=" + age +
", name='" + name + '\'' +
", name1='" + name1 + '\'' +
", name2='" + name2 + '\'' +
", password='" + password + '\'' +
'}';
}
}
我们修改所有字段不被transient修饰,只使用static、和final来修饰,我们来看看日志:
08-11 03:47:05.222 6872-6872/com.mark.java D/MainActivity: Write : User{bool=true, l=1234567890, d=1.23456789E8, f=1.2345679E8, s=123, b=1, age=20, name='Mark', name1='Mark', name2='Mark', password='coder'}
08-11 03:47:08.870 6872-6872/com.mark.java D/MainActivity: Read : User{bool=true, l=1234567890, d=1.23456789E8, f=1.2345679E8, s=123, b=1, age=20, name='Mark', name1='Mark', name2='Mark', password='coder'}
从日志中我们可以看出,static、和final单独修饰字段或者组合修饰字段都不会影响序列化和反序列化操作。接下来我们将上面User类的所有字段都加上transient关键字,如下:
private transient final boolean bool = true;
private transient final long l = 1234567890L;
private transient static double d = 123456789.0D;
private transient static float f = 123456789.0F;
private transient short s = 123;
private transient byte b = 1;
private transient int age = 20;
private transient static String name;
private transient final String name1 = "Mark";
private transient static final String name2 = "Mark";
private transient String password;
运行日志如下:
08-11 03:52:30.773 11948-11948/com.mark.java D/MainActivity: Write : User{bool=true, l=1234567890, d=1.23456789E8, f=1.2345679E8, s=123, b=1, age=20, name='Mark', name1='Mark', name2='Mark', password='coder'}
08-11 03:52:37.911 11948-11948/com.mark.java D/MainActivity: Read : User{bool=true, l=1234567890, d=1.23456789E8, f=1.2345679E8, s=0, b=0, age=0, name='Mark', name1='Mark', name2='Mark', password='null'}
从日志可以看出只有b、age、password的transient起了作用。恰巧这三个字段没有static、final修饰,故我们可以得出结论:
static、final修饰的字段在使用transient修饰,则transient关键字不起作用,即static、final修饰的字段可以正常进行序列化和反序列化。
1)一旦变量被transient修饰,变量将不再是对象持久化的一部分,该变量内容在序列化后无法获得访问。
2)transient关键字只能修饰变量,而不能修饰方法和类。注意,本地变量是不能被transient关键字修饰的。变量如果是用户自定义类变量,则该类需要实现Serializable接口。
3)被transient关键字修饰的变量不再能被序列化,一个静态变量不管是否被transient修饰,均不能被序列化。
在Android中要序列化可以实现Serializable接口或者Parcelable接口。以上所有测试都是基于Serializable接口,对于Parcelable不成立。因为Parcelable接口序列化过程是我们自己控制的,而Serializable接口序列化过程是系统控制的,所以以上测试和结论都是相对Serializable接口序列化而言。
如果实现Parcelable接口进行序列化则transient就不一定起作用了。有兴趣的小伙伴可以动手实践一下。