Java transient 关键字是干啥的
百度百科的解释:
Java语言的关键字,变量修饰符,如果用transient声明一个实例变量,当对象存储时,它的值不需要维持。换句话来说就是,用transient关键字标记的成员变量不参与序列化过程。
作用:
Java的serialization提供了一种持久化对象实例的机制。当持久化对象时,可能有一个特殊的对象数据成员,我们不想用serialization机制来保存它。为了在一个特定对象的一个域上关闭serialization,可以在这个域前加上关键字transient。当一个对象被序列化的时候,transient型变量的值不包括在序列化的表示中,然而非transient型的变量是被包括进去的。
编码试验加以证明:
1.自定义类(为了方便,我直接在main方法所在类中添加的一个静态属性类,或者叫成员类)
publicstaticclassTransientTestimplementsSerializable{ privatestaticfinallongserialVersionUID=233858934995755239L; privateStringname1; privatetransientStringname2; publicTransientTest(Stringname1,Stringname2){ this.name1=name1; this.name2=name2; } publicStringtoString(){ returnString.format("TransientTest.toString():name1=%s,name2=%s",name1,name2); } }
2.写一个测试方法
publicstaticvoidtestTransient(){ Stringname1="常规属性",name2="transient修饰的属性"; TransientTesttest=newTransientTest(name1,name2); System.out.println("序列化前:"+test.toString()); ObjectOutputStreamoutStream; ObjectInputStreaminStream; StringfilePath="D:/test/object/TransientTest.obj"; try{ outStream=newObjectOutputStream(newFileOutputStream(filePath)); outStream.writeObject(test); inStream=newObjectInputStream(newFileInputStream(filePath)); TransientTestreadObject=(TransientTest)inStream.readObject(); System.out.println("序列化后:"+readObject.toString()); }catch(IOExceptione){ //TODOAuto-generatedcatchblock e.printStackTrace(); }catch(ClassNotFoundExceptione){ //TODOAuto-generatedcatchblock e.printStackTrace(); } }
3.在main方法中执行,然后可以看到得到的结果为
印证了上面所讲的“用transient关键字标记的成员变量不参与序列化过程”。
用二进制查看器打开这个文件也可以看到,数据中只有name1,没有name2。(请忽略乱码问题,这个不是重点哈。)
延伸:
在查看JDK源码的时候会发现很多地方都会加上transient关键字来修饰一些属性,那究竟是出于什么考虑才这么做呢?
我觉得,应该是为了节约磁盘空间,避免造成不必要的浪费吧。
以ArrayList中的transientObject[]elementData为例,这个成员变量的注释为:
翻译出来就是:
/** *存储ArrayList元素的数组缓冲区。 *ArrayList的容量是这个数组缓冲区的长度。任何 *带有elementData的空ArrayList==DEFAULTCAPACITY_EMPTY_ELEMENTDATA *当添加第一个元素时,将被扩展到DEFAULT_CAPACITY。 */
这个缓冲区的容量实际上并不是ArrayList的容量,因为其实际上会预留一些空间,当空间不足时还会扩容,为减少浪费,因此在序列化时不会按照默认算法将这个成员变量写入磁盘。而是写了个writeObject方法,序列化时会调用这个方法将其持久化,在反序列化是,调用readObject,将其恢复出来。
这2个方法为:
参考ArrayList,在上面的TransientTest中添加2个方法,见代码:
publicstaticclassTransientTestimplementsSerializable{ privatestaticfinallongserialVersionUID=233858934995755239L; privateStringname1; privatetransientStringname2; publicTransientTest(Stringname1,Stringname2){ this.name1=name1; this.name2=name2; } publicStringtoString(){ returnString.format("TransientTest.toString():name1=%s,name2=%s",name1,name2); } privatevoidwriteObject(java.io.ObjectOutputStreams)throwsjava.io.IOException{ s.defaultWriteObject(); s.writeObject(name2); } privatevoidreadObject(java.io.ObjectInputStreams)throwsjava.io.IOException,ClassNotFoundException{ s.defaultReadObject(); name2=String.valueOf(s.readObject()); } }
然后在main方法中执行testTransient(),此时得到的结果是:
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持毛票票。
声明:本文内容来源于网络,版权归原作者所有,内容由互联网用户自发贡献自行上传,本网站不拥有所有权,未作人工编辑处理,也不承担相关法律责任。如果您发现有涉嫌版权的内容,欢迎发送邮件至:czq8825#qq.com(发邮件时,请将#更换为@)进行举报,并提供相关证据,一经查实,本站将立刻删除涉嫌侵权内容。