前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Java中transient关键字的作用

Java中transient关键字的作用

作者头像
搜云库技术团队
发布2019-10-18 01:55:23
9630
发布2019-10-18 01:55:23
举报
文章被收录于专栏:搜云库技术团队

先解释下Java中的对象序列化

在讨论transient之前,有必要先搞清楚Java中序列化的含义;

Java中对象的序列化指的是将对象转换成以字节序列的形式来表示,这些字节序列包含了对象的数据和信息,一个序列化后的对象可以被写到数据库或文件中,也可用于网络传输,一般当我们使用缓存cache(内存空间不够有可能会本地存储到硬盘)或远程调用rpc(网络传输)的时候,经常需要让我们的实体类实现Serializable接口,目的就是为了让其可序列化。

当然,序列化后的最终目的是为了反序列化,恢复成原先的Java对象,要不然序列化后干嘛呢,所以序列化后的字节序列都是可以恢复成Java对象的,这个过程就是反序列化。

关于transient关键字

Java中transient关键字的作用,简单地说,就是让某些被修饰的成员属性变量不被序列化,这一看好像很好理解,就是不被序列化,那么什么情况下,一个对象的某些字段不需要被序列化呢?如果有如下情况,可以考虑使用关键字transient修饰:

1、类中的字段值可以根据其它字段推导出来,如一个长方形类有三个属性:长度、宽度、面积(示例而已,一般不会这样设计),那么在序列化的时候,面积这个属性就没必要被序列化了;

2、其它,看具体业务需求吧,哪些字段不想被序列化;

PS,记得之前看HashMap源码的时候,发现有个字段是用transient修饰的,我觉得还是有道理的,确实没必要对这个modCount字段进行序列化,因为没有意义,modCount主要用于判断HashMap是否被修改(像put、remove操作的时候,modCount都会自增),对于这种变量,一开始可以为任何值,0当然也是可以(new出来、反序列化出来、或者克隆clone出来的时候都是为0的),没必要持久化其值。

代码语言:javascript
复制
/**
* The number of times this HashMap has been structurally modified
* Structural modifications are those that change the number of mappings in
* the HashMap or otherwise modify its internal structure (e.g.,
* rehash).  This field is used to make iterators on Collection-views of
* the HashMap fail-fast.  (See ConcurrentModificationException).
*/
transient int modCount;

最后,为什么要不被序列化呢,主要是为了节省存储空间,其它的感觉没啥好处,可能还有坏处(有些字段可能需要重新计算,初始化什么的),总的来说,利大于弊。

举个例子

仅仅是示例,具体使用请根据实际情况:

代码语言:javascript
复制
package com.ecej.esmart.workform.queue;
import com.ecej.esmart.core.annotation.Log;
import com.ecej.esmart.workform.dao.EsmartBaseDao;
import com.ecej.nove.utils.common.BaseUtils;
import org.apache.commons.lang.StringUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.amqp.rabbit.annotation.RabbitHandler;
import org.springframework.amqp.rabbit.annotation.RabbitListener;
import org.springframework.stereotype.Component;
import javax.annotation.Resource;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
/**
 * 删除周期订单虚拟档期表
 */
@Component
@RabbitListener(queues = "${esmart.del.cycle.order.schedule}")
public class ReceiverDelCycleOrderSchedule {
    private static final Logger LOGGER = LoggerFactory.getLogger(ReceiverDelCycleOrderSchedule.class);
    @Resource
    private EsmartBaseDao esmartBaseDao;
    @Log("删除周期订单虚拟档期表")
    @RabbitHandler
    public void delCycleOrderSchedule(String parentOrderNo) {
        try {
            LOGGER.info("监听到的消息:{}", parentOrderNo);
            if (StringUtils.isNotEmpty(parentOrderNo)) {
                Map<String, Object> objectMap = new HashMap<>();
                objectMap.put("parentOrderNo", parentOrderNo);
                esmartBaseDao.delete(BaseUtils.makeClazzPath(com.ecej.esmart.workform.ecejservice.po.WorkOrderPo.class, "delCycleOrderSchedule"), objectMap);
            }
        } catch (Exception e) {
            LOGGER.error(e.getMessage(), e);
        }
    }
}

结果打印(达到目的,节省存储空间,成功恢复成原始对象):

1.原始对象

width : 3 height : 4 area : 12

2.反序列化后的对象

width : 3 height : 4 area : null

3.恢复成原始对象

width : 3 height : 4 area : 12

参考资料

http://stackoverflow.com/questions/910374/why-does-java-have-transient-fields

往期精彩文章

本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2018-10-10,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 架构师技术栈 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
相关产品与服务
文件存储
文件存储(Cloud File Storage,CFS)为您提供安全可靠、可扩展的共享文件存储服务。文件存储可与腾讯云服务器、容器服务、批量计算等服务搭配使用,为多个计算节点提供容量和性能可弹性扩展的高性能共享存储。腾讯云文件存储的管理界面简单、易使用,可实现对现有应用的无缝集成;按实际用量付费,为您节约成本,简化 IT 运维工作。
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档