transient关键字标记成员变量使其在序列化时被跳过,仅对实现Serializable接口的类生效;static+transient冗余,final transient字段反序列化后按默认值初始化;适用于敏感数据、临时状态等场景。

transient 关键字用于标记类的成员变量,使其在对象序列化时被跳过——不写入字节流,也不参与反序列化恢复。
它只对序列化起作用
只有当类实现了 Serializable 接口,并使用 ObjectOutputStream 进行序列化时,transient 才生效。它不影响普通赋值、方法调用或内存中的运行状态。
- 未实现 Serializable 的类,即使加了 transient,也根本不会触发序列化逻辑
- static 变量本身就不参与序列化,所以 static + transient 是冗余写法,JVM 忽略 transient
- final 变量若被 transient 修饰,仍不序列化;但反序列化后会按 final 规则初始化(如 String 类型为 null,int 为 0)
典型适用场景
核心原则:该字段不该持久化、不该跨网络传输、或无法安全保存。
- 敏感数据:密码、token、密钥等,避免明文落盘或被中间节点截获
- 临时/派生状态:比如缓存计算结果、UI 组件引用、线程局部变量、数据库连接句柄等
- 不可序列化类型:成员变量类型未实现 Serializable(如某些第三方类、InputStream),又不想改其源码时,用 transient 跳过可避免 NotSerializableException
反序列化后的值怎么处理
transient 字段在反序列化后不会保留原值,而是直接赋予对应类型的默认值:
立即学习“Java免费学习笔记(深入)”;
- 基本类型 → 0(int/long)、false(boolean)、\u0000(char)
- 引用类型 → null
- 数组 → null(不是空数组)
如果需要自定义反序列化行为(例如重置某个 transient 字段),可实现 readObject 方法,在其中手动赋值。
和 Externalizable 的区别
Serializable 是“默认全量序列化,用 transient 黑名单排除”;Externalizable 则是“默认不序列化任何字段,必须在 writeExternal/readExternal 中显式写出”。后者更灵活,但开发成本高,且容易遗漏字段导致兼容问题。
transient 在 Externalizable 中依然有效——如果你在 writeExternal 中没写某个 transient 字段,它自然不会出现;但如果你写了,JVM 不会阻止,因为 Externalizable 完全由你控制。










