作为java中的一个关键字,tranisent用的并不是很多,但是在某些关键场合,却又起着极为重要的作用,因此有必要对它进行一些必要的了解。

一、定义:声明不用序列化的成员域。(源自百度百科)

二、作用:根据tranisent关键字的定义,我们可以很容易的归纳出它的作用,那就是修饰变量,使之不能成为对象持久化的一部分,当然,对于tranisent修饰的这个变量,也是有要求的,这个变量不能是本地变量,如果是用户自定义的变量,那么拥有这个变量的类必须实现Serializable接口;根据之前对static和final的描述,我们知道它们除了可以修饰变量之外,还可以修饰方法和类,但是对于tranisent来说,它有且仅有可以修饰变量的作用,对于方法和类,它是不允许修饰的。

1.修饰变量:对于tranisent修饰的变量,是不允许被序列化的。

  1. import java.io.FileInputStream;
  2. import java.io.FileOutputStream;
  3. import java.io.ObjectInputStream;
  4. import java.io.ObjectOutputStream;
  5. import java.io.Serializable;
  6.  
  7. public class Rectangle implements Serializable {
  8. private static final long serialVersionUID = 1710022455003682613L;
  9. private Integer width;
  10. private Integer height;
  11. private transient Integer area;
  12.  
  13. public Rectangle (Integer width, Integer height){
  14. this.width = width;
  15. this.height = height;
  16. this.area = width * height;
  17. }
  18.  
  19. public void setArea(){
  20. this.area = this.width * this.height;
  21. }
  22.  
  23. @Override
  24. public String toString(){
  25. StringBuffer sb = new StringBuffer(40);
  26. sb.append("width : ");
  27. sb.append(this.width);
  28. sb.append("\nheight : ");
  29. sb.append(this.height);
  30. sb.append("\narea : ");
  31. sb.append(this.area);
  32. return sb.toString();
  33. }
  34. }
  35.  
  36. public class TransientExample {
  37. public static void main(String args[]) throws Exception {
  38. Rectangle rectangle = new Rectangle(3,4);
  39. System.out.println("1.原始对象\n"+rectangle);
  40. ObjectOutputStream o = new ObjectOutputStream(new FileOutputStream("rectangle"));
  41. // 往流写入对象
  42. o.writeObject(rectangle);
  43. o.close();
  44.  
  45. // 从流读取对象
  46. ObjectInputStream in = new ObjectInputStream(new FileInputStream("rectangle"));
  47. Rectangle rectangle1 = (Rectangle)in.readObject();
  48. System.out.println("2.反序列化后的对象\n"+rectangle1);
  49. rectangle1.setArea();
  50. System.out.println("3.恢复成原始对象\n"+rectangle1);
  51. in.close();
  52. }
  53. }

输出结果:

  1. 1.原始对象
  2. width : 3
  3. height : 4
  4. area : 12
  5. 2.反序列化后的对象
  6. width : 3
  7. height : 4
  8. area : null
  9. 3.恢复成原始对象
  10. width : 3
  11. height : 4
  12. area : 12

注:被transient关键字修饰的变量不再能被序列化,一个静态变量不管是否被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. public class Person implements Externalizable {
  12.  
  13. private transient String content = "是的,我会被序列化";
  14.  
  15. @Override
  16. public void writeExternal(ObjectOutput out) throws IOException {
  17. out.writeObject(content);
  18. }
  19.  
  20. @Override
  21. public void readExternal(ObjectInput in) throws IOException,
  22. ClassNotFoundException {
  23. content = (String) in.readObject();
  24. }
  25.  
  26. public static void main(String[] args) throws Exception {
  27. Person person = new Person();
  28. ObjectOutput out = new ObjectOutputStream(new FileOutputStream(
  29. new File("test")));
  30. out.writeObject(person);
  31.  
  32. ObjectInput in = new ObjectInputStream(new FileInputStream(new File(
  33. "test")));
  34. person = (Person) in.readObject();
  35. System.out.println(person.content);
  36.  
  37. out.close();
  38. in.close();
  39. }
  40. }

输出结果:

  1. 是的,我会被序列化

十分意外的是,被transient关键字修饰的变量竟然也可以序列化了,这又是为什么呢?原来之前我们所说的,全都是建立在该类是实现Serializable接口的,但是我们知道,在java中,对象的序列化是可以通过两种接口来实现的,一个是Serializable接口,另一个就是上图中所展示的Externalizable接口,至于二者的具体区别,这里不详细诉说了,但是实现Serializable接口,所有的序列化都是自动进行的,实现Externalizable接口则不同,需要我们在writeExternal方法中进行手动序列化,因此这个时候transient就不能够起到什么作用了,这也是为什么在上面的例子中变量content能够被序列化的原因。

总结:虽然transient关键字不是那么常用,但是却不代表它不重要,这里我就简单的小结了一下,如果大家还有什么好的建议,欢迎在评论区留言。

参照:http://www.cnblogs.com/chenpi/

java关键字tranisent用法详解的更多相关文章

  1. java关键字static用法详解

    java中有53个关键字,其中包含2个保留字,这篇文章主要介绍一下static这个关键字. static在java中算是一个比较常见的关键字,有着多种用法,因此很有必要好好地了解一番. 一.定义 st ...

  2. java关键字volatile用法详解

    volatile关键字想必大家都不陌生,在java 5之前有着挺大的争议,在java 5之后才逐渐被大家接受,同时作为java的关键字之一,其作用自然是不可小觑的,要知道它是java.util.con ...

  3. java关键字final用法详解

    final关键字在java中也是属于比较常用的一种,因此也算得上是一个比较重要的关键字,有必要对它进行深入的学习. 一.定义:用来说明最终属性,表明一个类不能派生出子类,或者成员方法不能被覆盖,或者成 ...

  4. Java关键字总结及详解

    Java关键字是Java的保留字,这些保留字不能用来作为常量.变量.类名.方法名及其他一切标识符的名称. 一.基本数据类型 Java中有八种基本数据类型,六种数字类型(四个整数型.六中浮点型),一种字 ...

  5. 【转】Java enum的用法详解

    用法一:常量 在JDK1.5 之前,我们定义常量都是: public static fianl.... .现在好了,有了枚举,可以把相关的常量分组到一个枚举类型里,而且枚举提供了比常量更多的方法. p ...

  6. Java enum的用法详解[转]

    Ref:http://www.cnblogs.com/happyPawpaw/archive/2013/04/09/3009553.html 用法一:常量 在JDK1.5 之前,我们定义常量都是: p ...

  7. 多线程java的concurrent用法详解(转载)

    我们都知道,在JDK1.5之前,Java中要进行业务并发时,通常需要有程序员独立完成代码实现,当然也有一些开源的框架提供了这些功能,但是这些依然没有JDK自带的功能使用起来方便.而当针对高质量Java ...

  8. Java enum的用法详解

    (转自:http://www.cnblogs.com/happyPawpaw/archive/2013/04/09/3009553.html) 用法一:常量 在JDK1.5 之前,我们定义常量都是: ...

  9. java的concurrent用法详解

    我们都知道,在JDK1.5之前,Java中要进行业务并发时,通常需要有程序员独立完成代码实现,当然也有一些开源的框架提供了这些功能,但是这些依然没有JDK自带的功能使用起来方便.而当针对高质量Java ...

随机推荐

  1. Java -> 构造器(构造方法)

    构造方法 我们对封装已经有了基本的了解,接下来我们来看一个新的问题,依然以Person为例,由于Person中的属性都被private了,外界无法直接访问属性,必须对外提供相应的set和get方法.当 ...

  2. Spring全家桶之spring boot(二)

    spring boot的两种配置文件: 虽然spring boot可以帮助我们进行一些配置项,但是有些内容还是需要开发者自己进行配置,因此spring boot提供了配置文件以供开发者配置.sprin ...

  3. 【题解】[SCOI2015]小凸玩矩阵

    题目链接 思路:题目要求变相解答一下,求出是否有n-k个数,不大于当前求的第k个数 而每一行每一列只能有一个数,就可以得到一个二分图的思路,边上的权值就是第i行第j列这个数的值 对于答案就是第k大的数 ...

  4. MyBatis缓存机制(一级缓存,二级缓存)

    一,MyBatis一级缓存(本地缓存) My Batis 一级缓存存在于 SqlSession 的生命周期中,是SqlSession级别的缓存.在操作数据库时需要构造SqlSession对象,在对象中 ...

  5. [前端进阶课] 构建自己的 webpack 知识体系

    webpack webpack 最出色的功能之一就是,除了 JavaScript,还可以通过 loader 引入任何其他类型的文件. Webpack 核心概念: Entry(入口):Webpack 执 ...

  6. deno+mongo实战踩坑记

    自从 deno 1.0 发布以来,有关 deno 的文章很多,大多数都是在讨论怎么安装 deno .deno 有哪些特点 .deno 和 node 有哪些异同.deno是不是 node 的替代品等.咱 ...

  7. 基于 abp vNext 和 .NET Core 开发博客项目 - 接入GitHub,用JWT保护你的API

    上一篇文章(https://www.cnblogs.com/meowv/p/12924859.html)再次把Swagger的使用进行了讲解,完成了对Swagger的分组.描述和开启小绿锁以进行身份的 ...

  8. poi--读取不同类型的excel表格

    要想根据不同类型excel表格获取其数据,就要先判断其表格类型 poi-api种方法: getCellType    public int getCellType()        Return th ...

  9. JVM中的垃圾收集

    引用计数(Reference Counting) 循环引用问题 标记­清除(Mark and Sweep) 内存池(Memory Pools) Eden 是内存中的一个区域, 用来分配新创建的对象 . ...

  10. 基于Unity实现油画风格的着色器

    // Upgrade NOTE: replaced 'mul(UNITY_MATRIX_MVP,*)' with 'UnityObjectToClipPos(*)' Shader "Cust ...