Java中在時(shí)間戳計(jì)算的過(guò)程中遇到的數(shù)據(jù)溢出問(wèn)題解決
背景
今天在跑定時(shí)任務(wù)的過(guò)程中,發(fā)現(xiàn)有一個(gè)任務(wù)在設(shè)置數(shù)據(jù)的查詢時(shí)間范圍異常,出現(xiàn)了開始時(shí)間戳比結(jié)束時(shí)間戳大的奇怪現(xiàn)象,計(jì)算時(shí)間戳的代碼大致如下。
package com.lingyejun.authenticator; public class IntegerTest { public static void main(String[] args) { long endTime = System.currentTimeMillis(); long startTime = endTime - 30 * 24 * 60 * 60 * 1000; System.out.println("end : " + endTime); System.out.println("start : " + startTime); } }
先放出結(jié)論:因?yàn)閖ava中整數(shù)默認(rèn)是int類型,在計(jì)算的過(guò)程中30 * 24 * 60 * 60 * 1000計(jì)算結(jié)果大于Integer.MAX_VALUE,所以出現(xiàn)了數(shù)據(jù)溢出,從而導(dǎo)致了計(jì)算結(jié)果不準(zhǔn)確的問(wèn)題。
驗(yàn)證
我們將上面的代碼稍稍改造一下,方便我們確認(rèn)定位問(wèn)題,調(diào)整后的代碼如下:
package com.lingyejun.authenticator; public class IntegerTest { public static long calcStartTime(long endTime, long minusMills) { System.out.println("end : " + endTime + " minus mills : " + minusMills); long startTime = endTime - minusMills; System.out.println("start: " + startTime); return startTime; } public static void main(String[] args) { long nowTime = System.currentTimeMillis(); long a = 30 * 24 * 60 * 60 * 1000; calcStartTime(nowTime, a); } }
結(jié)果如下:
end : 1560869539864 minus mills : -1702967296
start: 1562572507160
這和我們的預(yù)期不一樣,因?yàn)?0 * 86400000 = 2592000000,但是計(jì)算出來(lái)卻是:-1702967296。
到這里想必大家都知道原因了,這是因?yàn)閖ava中整數(shù)的默認(rèn)類型是整型int,而int的最大值是2147483647,
在代碼中java是先計(jì)算右值,再賦值給long變量的。在計(jì)算右值的過(guò)程中(int型相乘)發(fā)生溢出,然后將溢出后截?cái)嗟闹蒂x給變量,導(dǎo)致了結(jié)果不準(zhǔn)確。
將代碼做一下小小的改動(dòng),再看一下。
package com.lingyejun.authenticator; public class IntegerTest { public static long calcStartTime(long endTime, long minusMills) { System.out.println("end : " + endTime + " minus mills : " + minusMills); long startTime = endTime - minusMills; System.out.println("start: " + startTime); return startTime; } public static void main(String[] args) { long nowTime = System.currentTimeMillis(); long a = 30 * 24 * 60 * 60 * 1000L; calcStartTime(nowTime, a); } }
結(jié)果為
end : 1560869539864 minus mills : 2592000000
start: 1558277539864
似乎這樣應(yīng)該就沒(méi)有什么問(wèn)題了,但是這樣就真的保險(xiǎn)了嗎,如果我要把30調(diào)整為24856(Integer.MAX_VALUE / 86400 = 24855),即改為:long a = 24856 * 24 * 60 * 60 * 1000L 那么同樣會(huì)出現(xiàn)溢出。
因?yàn)閖ava的運(yùn)算規(guī)則從左到右,再與最后一個(gè)long型的1000相乘之前就已經(jīng)溢出,所以結(jié)果也不對(duì),正確的方式應(yīng)該如下:long a = 24856L * 24 * 60 * 60 * 1000。
package com.lingyejun.authenticator; public class IntegerTest { public static long calcStartTime(long endTime, long minusMills) { System.out.println("end : " + endTime + " minus mills : " + minusMills); long startTime = endTime - minusMills; System.out.println("start: " + startTime); return startTime; } public static void main(String[] args) { long a = 30L * 24 * 60 * 60 * 1000; calcStartTime(nowTime, a); } }
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
SpringBoot+BootStrap多文件上傳到本地實(shí)例
這篇文章主要介紹了SpringBoot+BootStrap多文件上傳到本地實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-03-03使用Vue+Spring Boot實(shí)現(xiàn)Excel上傳功能
這篇文章主要介紹了使用Vue+Spring Boot實(shí)現(xiàn)Excel上傳,需要的朋友可以參考下2018-11-11Java利用VLC開發(fā)簡(jiǎn)易視屏播放器功能
這篇文章主要介紹了Java利用VLC開發(fā)簡(jiǎn)易視屏播放器,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-08-08???????Spring多租戶數(shù)據(jù)源管理 AbstractRoutingDataSource
本文技術(shù)了???????Spring多租戶數(shù)據(jù)源管理 AbstractRoutingDataSource,下文詳細(xì)內(nèi)容介紹,需要的小伙伴可以參考一下2022-05-05Flowable數(shù)據(jù)庫(kù)表分類及數(shù)據(jù)字典解析
這篇文章主要介紹了Flowable數(shù)據(jù)庫(kù)表分類及數(shù)據(jù)字典解析,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-11-11Java實(shí)現(xiàn)KFC點(diǎn)餐系統(tǒng)過(guò)程解析
這篇文章主要介紹了Java實(shí)現(xiàn)KFC點(diǎn)餐系統(tǒng)過(guò)程解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-10-10Java中Spring Boot支付寶掃碼支付及支付回調(diào)的實(shí)現(xiàn)代碼
這篇文章主要介紹了Java中Spring Boot支付寶掃碼支付及支付回調(diào)的實(shí)現(xiàn)代碼,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-02-02Java實(shí)現(xiàn)雪花算法(snowflake)
這篇文章主要介紹了Java實(shí)現(xiàn)雪花算法(snowflake),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-08-08Mybatis實(shí)現(xiàn)查詢相冊(cè)數(shù)據(jù)列表流程講解
這篇文章主要介紹了Mybatis實(shí)現(xiàn)查詢相冊(cè)數(shù)據(jù)列表流程,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)吧2022-12-12