百度360必应搜狗淘宝本站头条
当前位置:网站首页 > IT精选 > 正文

java开发:每日bug001(慎用对象的引用)

ccwork 2025-03-14 15:29 9 浏览

【Java江湖,bug(不拘)一格】

嘿,各位编程大侠、代码小侠们,欢迎来到我们的神秘小岛——“Java每日虫洞探险”!在这里,我们将一起记录并解决那些奇奇怪怪的bug。

案例分享:

我曾遇到这样一个场景:需要遍历一个对象集合,集合中包含月初和月末的日期对象Calendar。我的任务是分天查询每个id对应一个月的数据,每个id分天查询完成后,再重置日期查询下一个id。

异常现象:

在结果文件中,只存在第一个id的月数据。

问题代码:

java

for (MonthRange monthRange : monthRanges) {
    executorService.submit(() -> {
        // 初始化日期
        Calendar finalStartTimeBeginOfDay = monthRange.getMonthStart();
        Calendar finalEndTimeBeginOfDay = monthRange.getMonthEnd();
 
        List shipIdList = shipIdGetter.getShipIds(finalStartTimeBeginOfDay);
        for (String shipId : shipIdList) {
            // 日期重置
            Calendar startTimeBeginOfDay = finalStartTimeBeginOfDay;
            Calendar endTimeBeginOfDay = finalEndTimeBeginOfDay;
 
            ExportPositionResult tempResult = new ExportPositionResult();
 
            while (CalendarUtil.compare(startTimeBeginOfDay, endTimeBeginOfDay) <= 0) {
                try {
                    // 根据id按天查询月数据
                    // ...(省略具体查询代码)
                } finally {
                    // 开始日期加1天
                    startTimeBeginOfDay.add(Calendar.DATE, 1);
                }
            }
            tempWriter.flush();
        }
    });
}

相信各位肯定已经发现问题了!

--------------------------------------------

经过仔细排查日志,我发现第一次循环后,日期对象已经被修改。后续的startTimeBeginOfDay引用的finalStartTimeBeginOfDay对象也已经改变,导致后续循环中的数据查询出错。

修改后代码:

java

for (MonthRange monthRange : monthRanges) {
    executorService.submit(() -> {
        // 初始化日期
        Calendar finalStartTimeBeginOfDay = monthRange.getMonthStart();
        Calendar finalEndTimeBeginOfDay = monthRange.getMonthEnd();
 
        List shipIdList = shipIdGetter.getShipIds(finalStartTimeBeginOfDay);
        for (String shipId : shipIdList) {
            // 单独克隆一份对象副本使用,避免操作原始对象
            Calendar startTimeBeginOfDay = (Calendar) finalStartTimeBeginOfDay.clone();
            Calendar endTimeBeginOfDay = (Calendar) finalEndTimeBeginOfDay.clone();
 
            ExportPositionResult tempResult = new ExportPositionResult();
 
            while (CalendarUtil.compare(startTimeBeginOfDay, endTimeBeginOfDay) <= 0) {
                try {
                    // 根据id按天查询月数据
                    // ...(省略具体查询代码)
                } finally {
                    // 按天加,此时操作的是克隆后的对象,不影响原始对象
                    startTimeBeginOfDay.add(Calendar.DATE, 1);
                }
            }
            tempWriter.flush();
        }
    });
}

完美搞定!

总结:

在Java中,Calendar是一个抽象类,其具体实现(如GregorianCalendar)是可变的。这意味着当创建一个Calendar对象并对其进行修改时,这个对象的内部状态会随之改变。

在原始代码中,finalStartTimeBeginOfDay和finalEndTimeBeginOfDay是通过方法monthRange.getMonthStart()和monthRange.getMonthEnd()获取的Calendar对象。这些对象代表了某个月的第一天和最后一天的开始时刻。然而,由于Calendar对象是可变的,当我们将这些对象赋值给循环中的startTimeBeginOfDay和endTimeBeginOfDay时,实际上引用的是同一个对象。因此,在循环中对startTimeBeginOfDay调用add(Calendar.DATE, 1)时,也会修改finalStartTimeBeginOfDay引用的对象,导致循环超出预期。

解决办法:

使用克隆方法创建一个新的Calendar实例。这个新实例的初始状态与原始实例相同,但它们是完全独立的对象。因此,对克隆后的对象所做的任何修改都不会影响到原始对象。

在平时的开发过程中,我们应该注意多个代码块、多线程中使用同一个对象引用时是否满足预期结果。特别是在使用可变对象时,要谨慎处理对象的共享和修改。

结束语:

今天的Bug故事就讲到这里。虽然每个Bug都像是一场小小的“灾难”,但正是这些挑战让我们的编程之旅充满了乐趣和成就感。好啦,明天见了各位!愿你每天都能轻松搞定Bug,享受编程带来的无限乐趣!

【今日bug迹,明日传奇身】

相关推荐

二十三、Java类与对象简介(java第十一章类和对象)

在Java编程语言中,类(Class)和对象(Object)是面向对象编程(OOP)的核心概念。描述类类是Java程序的基本组成单元,是对象的模板。类定义了对象的属性和方法。属性是对象的状态信息,而方...

设计模式-结构型-代理模式(proxy)

1.概念需要给对象提供一个代理以控制对该对象的访问,访问对象不适合或者不能直接引用目标对象,代理对象作为访问对象和目标对象之间的中介;根据代理类生成时机不同,分为静态代理和动态代理;静态代理代理类在编...

深度解析设计模式七大原则之——里氏替换原则

临近端午节,各位读者,你们假期行程安排好了吗?“菜鸟”已经做好决定了,谁都不能阻拦(产品经理也不行),“菜鸟”要好好在家休息三天。最近实在是太累了,一直在疯狂的加班。好了好了言归正传,开始我们的正文。...

Java代理模式详解:智能中介的编程艺术

一、生活场景中的代理思维想象您要租房子,但不想直接与房东打交道,这时房产中介就发挥作用了:1.中介帮您筛选房源(访问控制)2.签约前验证房东资质(预处理)3.协助办理合同手续(功能增强)4.处...

哪个创意最能打动你? 为你欣赏的“创意之星”投一票

这一期的《超级课堂·暑期特别活动》将评出5位“创意之星”,获得价值2000元的奖品。本期我们选登了部分中小学生在昙华林留下的创意作品,欢迎为最能打动你的作品投上一票。大众评审目前采取微信投票:扫描二维...

Netty基础—6.Netty实现RPC服务(netty reactor)

大纲1.RPC的相关概念2.RPC服务调用端动态代理实现3.Netty客户端之RPC远程调用过程分析4.RPC网络通信中的编码解码器5.Netty服务端之RPC服务提供端的处理6.RPC服务调用端实现...

静态代理和动态代理(静态代理和动态代理的优缺点)

1.什么是代理很多人肯定听过和看到过飞机票代理点,火车票代理点。那这些代理点干得事情就是帮航空公司,火车站出售火车票的工作。它们算是一个中间商。实际的服务不是由它们提供。而是由真正的服务商提供。通过这...

Java反射机制与Spring动态代理深度解析

一、Java反射机制原理剖析1.1反射的本质与实现基础Java反射(Reflection)是Java语言的核心特性,允许程序在运行时:动态加载类获取类结构元数据操作类属性和方法关键技术支撑:java...

Java 代理模式详解(java代理原理)

1.代理模式代理模式是一种比较好理解的设计模式。简单来说就是我们使用代理对象来代替对真实对象(realobject)的访问,这样就可以在不修改原目标对象的前提下,提供额外的功能操作,扩展目标对象...

SpringBoot全局异常处理:如何优雅应对多系统多格式错误响应需求

SpringBoot全局异常处理:如何优雅应对多系统多格式错误响应需求引言部分在微服务架构中,你是否曾为处理不同外部系统的异常响应而头痛?A系统要求返回JSON格式的:{code:1001,mes...

3分钟吃透代理技术!(代理一般都是怎么做)

最近有学员问了我一些问题,什么是代理,又该在什么地方使用。结合之前的讨论,这篇文章我们一起细致的讲解一下关于代理的一些问题。在Java中,代理通常分为两类:静态代理动态代理两者技术实现是不一样的,...

苏州网络维护 | 学习网络维护,从哪入手

我们想要学习网络维护,从哪入手呢?先带大家了解下网络维护1.培养基础知识:建立对计算机网络基本原理的理解。学习计算机网络的基础概念,如IP地址、子网掩码、路由器、交换机、协议等。2.学习网络技术:深入...

CAD如何快速一键编号?(cad如何一次性全部编号)

cad一键自动编号。·第一步,在命令行数abh空格。·第二步,打开自动编号对话框,选择用默认的图言编号,编号文字的高度根据图纸的大小设置零点八。当然如果图纸很大,设置比如十一百的有可能数字编号,这点很...

职场新人必知的10个高效工作法,助你快速升职加薪

初入职场,面对繁杂的工作任务和陌生的职场环境,如何才能快速适应并脱颖而出?以下是10个高效工作法,帮助职场新人提升工作效率,快速实现升职加薪的目标。---1.制定每日工作计划-推荐理由:每天开始...

《魔导英雄传说》新手攻略,快速升级,礼包码,最强阵容排列

《魔导英雄传说》新手攻略一、武将培养指南英雄选择与培养:英雄品质分为SSR、SR、A,优先培养SSR英雄,如张飞、孙尚香等核心英雄阵营搭配:同阵营上阵英雄越多,战力加成越高,建议优先培养同一阵营的英雄...