文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

Java中在时间戳计算的过程中遇到的数据溢出问题

2024-12-02 12:23

关注

今天在跑定时任务的过程中,发现有一个任务在设置数据的查询时间范围异常,出现了开始时间戳比结束时间戳大的奇怪现象,计算时间戳的代码大致如下。

  1. package com.lingyejun.authenticator; 
  2.  
  3. public class IntegerTest { 
  4.  
  5.     public static void main(String[] args) { 
  6.         long endTime = System.currentTimeMillis(); 
  7.         long startTime = endTime - 30 * 24 * 60 * 60 * 1000; 
  8.  
  9.         System.out.println("end   : " + endTime); 
  10.         System.out.println("start : " + startTime); 
  11.     } 

先放出结论:因为java中整数默认是int类型,在计算的过程中30 * 24 * 60 * 60 * 1000计算结果大于Integer.MAX_VALUE,所以出现了数据溢出,从而导致了计算结果不准确的问题。

验证

我们将上面的代码稍稍改造一下,方便我们确认定位问题,调整后的代码如下:

  1. package com.lingyejun.authenticator; 
  2.  
  3. public class IntegerTest { 
  4.  
  5.     public static long calcStartTime(long endTime, long minusMills) { 
  6.         System.out.println("end  : " + endTime + " minus mills : " + minusMills); 
  7.         long startTime = endTime - minusMills; 
  8.         System.out.println("start: " + startTime); 
  9.         return startTime; 
  10.     } 
  11.  
  12.     public static void main(String[] args) { 
  13.         long nowTime = System.currentTimeMillis(); 
  14.         long a = 30 * 24 * 60 * 60 * 1000; 
  15.         calcStartTime(nowTime, a); 
  16.     } 
  17. }   

结果如下:

  1. end  : 1560869539864 minus mills : -1702967296 
  2. start: 1562572507160 

这和我们的预期不一样,因为30 * 86400000 = 2592000000,但是计算出来却是:-1702967296。

到这里想必大家都知道原因了,这是因为java中整数的默认类型是整型int,而int的最大值是2147483647,

在代码中java是先计算右值,再赋值给long变量的。在计算右值的过程中(int型相乘)发生溢出,然后将溢出后截断的值赋给变量,导致了结果不准确。

将代码做一下小小的改动,再看一下。

  1. package com.lingyejun.authenticator; 
  2.  
  3. public class IntegerTest { 
  4.  
  5.     public static long calcStartTime(long endTime, long minusMills) { 
  6.         System.out.println("end  : " + endTime + " minus mills : " + minusMills); 
  7.         long startTime = endTime - minusMills; 
  8.         System.out.println("start: " + startTime); 
  9.         return startTime; 
  10.     } 
  11.  
  12.     public static void main(String[] args) { 
  13.         long nowTime = System.currentTimeMillis(); 
  14.         long a = 30 * 24 * 60 * 60 * 1000L; 
  15.         calcStartTime(nowTime, a); 
  16.     } 

结果为

  1. end  : 1560869539864 minus mills : 2592000000 
  2. start: 1558277539864 

似乎这样应该就没有什么问题了,但是这样就真的保险了吗,如果我要把30调整为24856(Integer.MAX_VALUE / 86400 = 24855),即改为:long a = 24856 * 24 * 60 * 60 * 1000L 那么同样会出现溢出。

因为java的运算规则从左到右,再与最后一个long型的1000相乘之前就已经溢出,所以结果也不对,正确的方式应该如下:long a = 24856L * 24 * 60 * 60 * 1000。

  1. package com.lingyejun.authenticator; 
  2.  
  3. public class IntegerTest { 
  4.  
  5.     public static long calcStartTime(long endTime, long minusMills) { 
  6.         System.out.println("end  : " + endTime + " minus mills : " + minusMills); 
  7.         long startTime = endTime - minusMills; 
  8.         System.out.println("start: " + startTime); 
  9.         return startTime; 
  10.     } 
  11.  
  12.     public static void main(String[] args) { 
  13.         long a = 30L * 24 * 60 * 60 * 1000; 
  14.         calcStartTime(nowTime, a); 
  15.     } 

 

来源:今日头条内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯