1. transient的作用及使用方法

我们都知道一个对象只要实现了Serilizable接口,这个对象就可以被序列化,java的这种序列化模式为开发者提供了很多便利,我们可以不必关系具体序列化的过程,只要这个类实现了Serilizable接口,这个类的所有属性和方法都会自动序列化。

然而在实际开发过程中,我们常常会遇到这样的问题,这个类的有些属性需要序列化,而其他属性不需要被序列化,打个比方,如果一个用户有一些敏感信息(如密码,银行卡号等),为了安全起见,不希望在网络操作(主要涉及到序列化操作,本地序列化缓存也适用)中被传输,这些信息对应的变量就可以加上transient关键字。换句话说,这个字段的生命周期仅存于调用者的内存中而不会写到磁盘里持久化。

总之,java 的transient关键字为我们提供了便利,你只需要实现Serilizable接口,将不需要序列化的属性前添加关键字transient,序列化对象的时候,这个属性就不会序列化到指定的目的地中。

  1. import java.io.FileInputStream;
  2. import java.io.FileNotFoundException;
  3. import java.io.FileOutputStream;
  4. import java.io.IOException;
  5. import java.io.ObjectInputStream;
  6. import java.io.ObjectOutputStream;
  7. import java.io.Serializable;
  8. /**
  9. * @description 使用transient关键字不序列化某个变量
  10. * 注意读取的时候,读取数据的顺序一定要和存放数据的顺序保持一致
  11. *
  12. * @author Alexia
  13. * @date 2013-10-15
  14. */
  15. public class TransientTest {
  16. public static void main(String[] args) {
  17. User user = new User();
  18. user.setUsername("Alexia");
  19. user.setPasswd("123456");
  20. System.out.println("read before Serializable: ");
  21. System.out.println("username: " + user.getUsername());
  22. System.err.println("password: " + user.getPasswd());
  23. try {
  24. ObjectOutputStream os = new ObjectOutputStream(
  25. new FileOutputStream("C:/user.txt"));
  26. os.writeObject(user); // 将User对象写进文件
  27. os.flush();
  28. os.close();
  29. } catch (FileNotFoundException e) {
  30. e.printStackTrace();
  31. } catch (IOException e) {
  32. e.printStackTrace();
  33. }
  34. try {
  35. ObjectInputStream is = new ObjectInputStream(new FileInputStream(
  36. "C:/user.txt"));
  37. user = (User) is.readObject(); // 从流中读取User的数据
  38. is.close();
  39. System.out.println("\nread after Serializable: ");
  40. System.out.println("username: " + user.getUsername());
  41. System.err.println("password: " + user.getPasswd());
  42. } catch (FileNotFoundException e) {
  43. e.printStackTrace();
  44. } catch (IOException e) {
  45. e.printStackTrace();
  46. } catch (ClassNotFoundException e) {
  47. e.printStackTrace();
  48. }
  49. }
  50. }
  51. class User implements Serializable {
  52. private static final long serialVersionUID = 8294180014912103005L;
  53. private String username;
  54. private transient String passwd;
  55. public String getUsername() {
  56. return username;
  57. }
  58. public void setUsername(String username) {
  59. this.username = username;
  60. }
  61. public String getPasswd() {
  62. return passwd;
  63. }
  64. public void setPasswd(String passwd) {
  65. this.passwd = passwd;
  66. }
  67. }

输出为:

  1. read before Serializable:
  2. username: Alexia
  3. password: 123456
  4. read after Serializable:
  5. username: Alexia
  6. password: null

密码字段为null,说明反序列化时根本没有从文件中获取到信息。

2. transient使用小结

1)一旦变量被transient修饰,变量将不再是对象持久化的一部分,该变量内容在序列化后无法获得访问。

2)transient关键字只能修饰变量,而不能修饰方法和类。注意,本地变量是不能被transient关键字修饰的。变量如果是用户自定义类变量,则该类需要实现Serializable接口。

3)被transient关键字修饰的变量不再能被序列化,一个静态变量不管是否被transient修饰,均不能被序列化

第三点可能有些人很迷惑,因为发现在User类中的username字段前加上static关键字后,程序运行结果依然不变,即static类型的username也读出来为“Alexia”了,这不与第三点说的矛盾吗?实际上是这样的:第三点确实没错(一个静态变量不管是否被transient修饰,均不能被序列化),反序列化后类中static型变量username的值为当前JVM中对应static变量的值,这个值是JVM中的不是反序列化得出的,不相信?好吧,下面我来证明:

  1. import java.io.FileInputStream;
  2. import java.io.FileNotFoundException;
  3. import java.io.FileOutputStream;
  4. import java.io.IOException;
  5. import java.io.ObjectInputStream;
  6. import java.io.ObjectOutputStream;
  7. import java.io.Serializable;
  8. /**
  9. * @description 使用transient关键字不序列化某个变量
  10. * 注意读取的时候,读取数据的顺序一定要和存放数据的顺序保持一致
  11. *
  12. * @author Alexia
  13. * @date 2013-10-15
  14. */
  15. public class TransientTest {
  16. public static void main(String[] args) {
  17. User user = new User();
  18. user.setUsername("Alexia");
  19. user.setPasswd("123456");
  20. System.out.println("read before Serializable: ");
  21. System.out.println("username: " + user.getUsername());
  22. System.err.println("password: " + user.getPasswd());
  23. try {
  24. ObjectOutputStream os = new ObjectOutputStream(
  25. new FileOutputStream("C:/user.txt"));
  26. os.writeObject(user); // 将User对象写进文件
  27. os.flush();
  28. os.close();
  29. } catch (FileNotFoundException e) {
  30. e.printStackTrace();
  31. } catch (IOException e) {
  32. e.printStackTrace();
  33. }
  34. try {
  35. // 在反序列化之前改变username的值
  36. User.username = "jmwang";
  37. ObjectInputStream is = new ObjectInputStream(new FileInputStream(
  38. "C:/user.txt"));
  39. user = (User) is.readObject(); // 从流中读取User的数据
  40. is.close();
  41. System.out.println("\nread after Serializable: ");
  42. System.out.println("username: " + user.getUsername());
  43. System.err.println("password: " + user.getPasswd());
  44. } catch (FileNotFoundException e) {
  45. e.printStackTrace();
  46. } catch (IOException e) {
  47. e.printStackTrace();
  48. } catch (ClassNotFoundException e) {
  49. e.printStackTrace();
  50. }
  51. }
  52. }
  53. class User implements Serializable {
  54. private static final long serialVersionUID = 8294180014912103005L;
  55. public static String username;
  56. private transient String passwd;
  57. public String getUsername() {
  58. return username;
  59. }
  60. public void setUsername(String username) {
  61. this.username = username;
  62. }
  63. public String getPasswd() {
  64. return passwd;
  65. }
  66. public void setPasswd(String passwd) {
  67. this.passwd = passwd;
  68. }
  69. }

运行结果为:

  1. read before Serializable:
  2. username: Alexia
  3. password: 123456
  4. read after Serializable:
  5. username: jmwang
  6. password: null

这说明反序列化后类中static型变量username的值为当前JVM中对应static变量的值,为修改后jmwang,而不是序列化时的值Alexia。

3. transient使用细节——被transient关键字修饰的变量真的不能被序列化吗?

思考下面的例子:

  1. import java.io.Externalizable;
  2. import java.io.File;
  3. import java.io.FileInputStream;
  4. import java.io.FileOutputStream;
  5. import java.io.IOException;
  6. import java.io.ObjectInput;
  7. import java.io.ObjectInputStream;
  8. import java.io.ObjectOutput;
  9. import java.io.ObjectOutputStream;
  10. /**
  11. * @descripiton Externalizable接口的使用
  12. *
  13. * @author Alexia
  14. * @date 2013-10-15
  15. *
  16. */
  17. public class ExternalizableTest implements Externalizable {
  18. private transient String content = "是的,我将会被序列化,不管我是否被transient关键字修饰";
  19. @Override
  20. public void writeExternal(ObjectOutput out) throws IOException {
  21. out.writeObject(content);
  22. }
  23. @Override
  24. public void readExternal(ObjectInput in) throws IOException,
  25. ClassNotFoundException {
  26. content = (String) in.readObject();
  27. }
  28. public static void main(String[] args) throws Exception {
  29. ExternalizableTest et = new ExternalizableTest();
  30. ObjectOutput out = new ObjectOutputStream(new FileOutputStream(
  31. new File("test")));
  32. out.writeObject(et);
  33. ObjectInput in = new ObjectInputStream(new FileInputStream(new File(
  34. "test")));
  35. et = (ExternalizableTest) in.readObject();
  36. System.out.println(et.content);
  37. out.close();
  38. in.close();
  39. }
  40. }

content变量会被序列化吗?好吧,我把答案都输出来了,是的,运行结果就是:

  1. 是的,我将会被序列化,不管我是否被transient关键字修饰

这是为什么呢,不是说类的变量被transient关键字修饰以后将不能序列化了吗?

我们知道在Java中,对象的序列化可以通过实现两种接口来实现,若实现的是Serializable接口,则所有的序列化将会自动进行,若实现的是Externalizable接口,则没有任何东西可以自动序列化,需要在writeExternal方法中进行手工指定所要序列化的变量,这与是否被transient修饰无关。因此第二个例子输出的是变量content初始化的内容,而不是null。

【Java基础】transient关键字的更多相关文章

  1. Java 序列化 transient关键字

    Java 序列化 transient关键字 @author 敏敏Alexia 转自:http://www.cnblogs.com/lanxuezaipiao/p/3369962.html 1. tra ...

  2. Java基础-synchronized关键字的用法(转载)

    synchronized--同步 顾名思义是用于同步互斥的作用的. 这里精简的记一下它的使用方法以及意义: 当synchronized修饰 this或者非静态方法或者是一个实例的时候,所同步的锁是加在 ...

  3. Java的transient关键字

    Java的transient关键字   Java 中的 transient 关键字被用来表示变量将不被序列化处理.那么在理解 transient 关键字之前,我们先了解下什么是序列化. 什么是序列化 ...

  4. Java基础之关键字,标识符,变量

    Java基础 首先,来看一下Java基础知识图解,以下便是在java学习中我们需要学习设计到的一些知识(当然不是很完全). 这些都是接下来在以后的学习中我们会学到的一些知识. 1 关键字 首次先来学习 ...

  5. java中transient关键字的作用

    Java有个特点就是序列化,简单地来说就是可以将这个类存储在物理空间(当然还是以文件的形式存在),那么当你从本地还原这个文件时,你可以将它转换为它本身.这可以极大地方便网络上的一些操作,但同时,因为涉 ...

  6. Java Volatile transient 关键字

    随笔-204  评论-134  文章-0  trackbacks-0   Volatile修饰的成员变量在每次被线程访问时,都强迫从主内存中重读该成员变量的值.而且,当成员变量发生变化时,强迫线程将变 ...

  7. Java的transient关键字(转)

    Volatile修饰的成员变量在每次被线程访问时,都强迫从主内存中重读该成员变量的值.而且,当成员变量发生变化时,强迫线程将变化值回写到主内存.这样在任何时刻,两个不同的线程总是看到某个成员变量的同一 ...

  8. java基础---->final关键字的使用

    这里介绍一些java基础关于final的使用,文字说明部分摘自java语言规范.心甘情愿这四个字,透着一股卑微,但也有藏不住的勇敢. Final关键字的说明 一.关于final变量规范说明 .A fi ...

  9. 干了这杯Java之transient关键字

    看源码的时候,发现transient这个关键字,不甚理解,查找资料发现:不被序列化 疑问: 静态变量是不是不被序列化? public class User implements Serializabl ...

  10. java基础Synchronized关键字之对象锁

    java中Synchronized关键字之对象锁    当有多个线程对一个共享数据进行操作时,需要注意多线程的安全问题. 多线程的同步机制对资源进行加锁,使得在同一个时间,只有一个线程可以进行操作,同 ...

随机推荐

  1. CSS 海盗船加载特效

    CSS 海盗船加载特效 <!DOCTYPE html> <html lang="en"> <head> <meta charset=

  2. 终论到底该不该写isDebugEnabled

    以前: 很多日志框架都不支持{}模板的写法(如Log4j1.X, Apache Commons Logging),于是只能通过字符串拼接来输出日志内容: log.debug("debug日志 ...

  3. 体验webhooks

    一.webhooks是什么 webhooks是一种实现在web api跟web service之间的发布订阅的轻量级的模式:当服务中心某个事件发生的时候,就会向订阅者发送一个POST请求形式的通知,这 ...

  4. WebSocket实现简易的FTP客户端

    WebScoket的简单应用,实现一个简易的FTP,即文件上传下载,可以查看上传人,下载次数,打开多个Web可以多人上传. 说在前面的话 文件传输协议(File Transfer Protocol,F ...

  5. Mysql - 如何决定用 datetime、timestamp、int 哪种类型存储时间戳?

    背景 数据表都很可能会有一两个字段需要保存日期时间数据,那应该用什么 Mysql 类型来保存呢? 前面讲过 datetime.timestamp.int 的方式来保存日期时间 如何存储 10位.13位 ...

  6. mybatis插入数据时处理为null的属性

    在做项目的时候,数据库中的所有字段被设置为全都不能为null,但是在我们开发过程中,插入一些记录的时候,实体类中的一些字段如果页面没有传入,则默认就会被设置为null,这样的话,在执行插入语句的时候, ...

  7. HTTP 缓存终极指南

    TL;DR 错误的缓存策略是如何抵消你所做的性能优化工作的. 缓存存在于客户端并且通过chrome或者其他抓包工具查看其状态信息. 客户端通过header中的各个字段做缓存的过期判断. 代理服务器上也 ...

  8. 巧用Python快速构建网页服务器

    经常做web开发,要调试一个网页,直接打开文件,用file模式显然是业余的. 但动辄要部署个IIS或APACHE站点,也确实太累,怎么办? 逐浪君此前有分享过通过http-server来构建快速的we ...

  9. java动态编译——tools.jar问题

    笔者在学习中写了一段简单的动态编译代码,但编译一直无法通过,起初认为受路径中存在汉字影响,修改路径后仍然没有解决.最终定位错误是:Java在进行动态编译的时候需要用到tools.jar资源包,若too ...

  10. puts()_C语言

    puts()函数用来向标准输出设备, scanf函数是格式输入函数,即按用户指定的格式从键盘上把数据输入到指定的变量之中. puts就是输出字符串啊.int puts(    const char* ...