1,transient的用途及使用方法
1,用途

我们知道,当一个对象实现了Serilizable接口,这个对象就可以被序列化,我们不关心其内在的原理,只需要了解这个类实现了Serilizable接口,这个类的所有属性和方法都会自动序列化。而在开发过程中,我们可能要求:当对象被序列化时(写入字节序列到目标文件)时,有些属性需要序列化,而其他属性不需要被序列化,打个比方,如果一个用户有一些敏感信息(如密码,银行卡号等),为了安全起见,不希望在网络操作(主要涉及到序列化操作,本地序列化缓存也适用)中被传输,这些信息对应的变量就可以加上transient关键字。换句话说,这个字段的生命周期仅存于调用者的内存中而不会写到磁盘里持久化。
所以,transient的用途在于:阻止实例中那些用此关键字声明的变量持久化;当对象被反序列化时(从源文件读取字节序列进行重构),这样的实例变量值不会被持久化和恢复。例如,当反序列化对象——数据流(例如,文件)可能不存在时,原因是你的对象中存在类型为java.io.InputStream的变量,序列化时这些变量引用的输入流无法被打开。
2,使用方法

序列化的时候,将不需要序列化的属性前添加关键字transient即可。
示例:

package newDay.day13;

import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.Serializable;

class UserInfo implements Serializable {  
    private static final long serialVersionUID = 996890129747019948L;  
    private String name;  
    private transient String psw;

public UserInfo(String name, String psw) {  
        this.name = name;  
        this.psw = psw;  
    }

public String toString() {  
        return "name=" + name + ", psw=" + psw;  
    }  
}  
public class TestTransient {
    public static void main(String[] args) {  
        UserInfo userInfo = new UserInfo("张三", "123456");  
        System.out.println(userInfo);  
        try {  
            // 序列化,被设置为transient的属性没有被序列化  
            ObjectOutputStream o = new ObjectOutputStream(new FileOutputStream("UserInfo.txt"));  
            o.writeObject(userInfo);  
            o.close();  
        } catch (Exception e) {  
            // TODO: handle exception  
            e.printStackTrace();  
        }  
        try {  
            // 重新读取内容  
            ObjectInputStream in = new ObjectInputStream(new FileInputStream("UserInfo.txt"));  
            UserInfo readUserInfo = (UserInfo) in.readObject();  
            //读取后psw的内容为null  
            System.out.println(readUserInfo.toString());  
        } catch (Exception e) {  
            // TODO: handle exception  
            e.printStackTrace();  
        }  
    }  
}

运行结果:

name=张三, psw=123456
name=张三, psw=null

密码字段为null,说明被标记为transient的属性在对象被序列化的时候不会被保存。
使用小结:

1,一旦变量被transient修饰,变量将不再是对象持久化的一部分,该变量内容在序列化后无法获得访问。
2,transient关键字只能修饰变量,而不能修饰方法和类。注意,本地变量是不能被transient关键字修饰的。变量如果是用户自定义类变量,则该类需要实现Serializable接口。
3,被transient关键字修饰的变量不再能被序列化,一个静态变量不管是否被transient修饰,均不能被序列化。
对于第三点,加上static之后,依然能把姓名输出。这是因为:反序列化后类中static型变量name的值为当前JVM中对应static变量的值,这个值是JVM中的不是反序列化得出的。下例可说明,其值时JVM中得到的而不是反序列化得到的:

package newDay.day13;

import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.Serializable;

class UserInfo implements Serializable {  
    private static final long serialVersionUID = 996890129747019948L;  
    private static String name;  
    private transient String psw;

public UserInfo(String name, String psw) {  
        this.name = name;  
        this.psw = psw;  
    }

public static String getName() {
        return name;
    }

public static void setName(String name) {
        UserInfo.name = name;
    }

public String getPsw() {
        return psw;
    }

public void setPsw(String psw) {
        this.psw = psw;
    }

public String toString() {  
        return "name=" + name + ", psw=" + psw;  
    }  
}  
public class TestTransient {
    public static void main(String[] args) {  
        UserInfo userInfo = new UserInfo("张三", "123456");
        System.out.println(userInfo);  
        try {  
            // 序列化,被设置为transient的属性没有被序列化  
            ObjectOutputStream o = new ObjectOutputStream(new FileOutputStream("UserInfo.txt"));  
            o.writeObject(userInfo);  
            o.close();  
        } catch (Exception e) {  
            // TODO: handle exception  
            e.printStackTrace();  
        }  
        try {  
            //在反序列化之前改变name的值
            userInfo.setName("hello");
            // 重新读取内容  
            ObjectInputStream in = new ObjectInputStream(new FileInputStream("UserInfo.txt"));  
            UserInfo readUserInfo = (UserInfo) in.readObject();  
            //读取后psw的内容为null  
            System.out.println(readUserInfo.toString());  
        } catch (Exception e) {  
            // TODO: handle exception  
            e.printStackTrace();  
        }  
    }  
}

运行结果:

name=张三, psw=123456
name=hello, psw=null

这说明反序列化后类中static型变量name的值为当前JVM中对应static变量的值,为修改后hello,而不是序列化时的值“张三”

原文链接:https://blog.csdn.net/u013207877/article/details/52572975

最新文章

  1. c#内部类的使用
  2. dock停靠管理器
  3. SQL Server中的索引结构与疑惑
  4. Sqlserver_判断该路径是否存在该文件
  5. 淘宝API开发(三)
  6. python学习第三天 --布尔类型
  7. 【转】简述什么是Web服务(Web Service)技术?
  8. Nginx也应用场景小结
  9. java调优(一)
  10. Dynamics 365 解决方案导出报错
  11. WPF 和 百度 eChart 交互
  12. 新鲜出炉的一套Java面试题
  13. 【转】python:让源码更安全之将py编译成so
  14. https://www.52pojie.cn/thread-688820-1-1.html
  15. SSIS 更新变量
  16. 使用apache的poi实现导入导出excel
  17. Gradle Goodness: Automatic Clean Tasks
  18. 权限认证与授权(Shrio 框架)
  19. 数据库的ACID特性详解
  20. HBase环境搭建随记

热门文章

  1. javascript Class.method vs Class.prototype.method(类方法和对象方法)
  2. Java之路---Day15(Collection类)
  3. Django--一对多表操作
  4. 为什么K8s会成为主流?
  5. JavaSE01:初始Java
  6. vue-cli2和3中的config
  7. 你与BAT只差这一套面试题
  8. 解决vue多次提交
  9. 一、MySQL基础知识
  10. pandas 之 特征工程