這篇文章主要介紹Java中如何解決在時間戳計算過程中遇到的數(shù)據(jù)溢出問題,文中介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們一定要看完!
建安網(wǎng)站制作公司哪家好,找創(chuàng)新互聯(lián)公司!從網(wǎng)頁設(shè)計、網(wǎng)站建設(shè)、微信開發(fā)、APP開發(fā)、響應(yīng)式網(wǎng)站開發(fā)等網(wǎng)站項目制作,到程序開發(fā),運(yùn)營維護(hù)。創(chuàng)新互聯(lián)公司從2013年成立到現(xiàn)在10年的時間,我們擁有了豐富的建站經(jīng)驗和運(yùn)維經(jīng)驗,來保證我們的工作的順利進(jìn)行。專注于網(wǎng)站建設(shè)就選創(chuàng)新互聯(lián)公司。
背景
在跑定時任務(wù)的過程中,發(fā)現(xiàn)有一個任務(wù)在設(shè)置數(shù)據(jù)的查詢時間范圍異常,出現(xiàn)了開始時間戳比結(jié)束時間戳大的奇怪現(xiàn)象,計算時間戳的代碼大致如下。
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é)論:因為java中整數(shù)默認(rèn)是int類型,在計算的過程中30 * 24 * 60 * 60 * 1000計算結(jié)果大于Integer.MAX_VALUE,所以出現(xiàn)了數(shù)據(jù)溢出,從而導(dǎo)致了計算結(jié)果不準(zhǔn)確的問題。
驗證
我們將上面的代碼稍稍改造一下,方便我們確認(rè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ù)期不一樣,因為30 * 86400000 = 2592000000,但是計算出來卻是:-1702967296。
到這里想必大家都知道原因了,這是因為java中整數(shù)的默認(rèn)類型是整型int,而int的最大值是2147483647,
在代碼中java是先計算右值,再賦值給long變量的。在計算右值的過程中(int型相乘)發(fā)生溢出,然后將溢出后截斷的值賦給變量,導(dǎo)致了結(jié)果不準(zhǔn)確。
將代碼做一下小小的改動,再看一下。
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)該就沒有什么問題了,但是這樣就真的保險了嗎,如果我要把30調(diào)整為24856(Integer.MAX_VALUE / 86400 = 24855),即改為:long a = 24856 * 24 * 60 * 60 * 1000L 那么同樣會出現(xiàn)溢出。
因為java的運(yùn)算規(guī)則從左到右,再與最后一個long型的1000相乘之前就已經(jīng)溢出,所以結(jié)果也不對,正確的方式應(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); } }
以上是“Java中如何解決在時間戳計算過程中遇到的數(shù)據(jù)溢出問題”這篇文章的所有內(nèi)容,感謝各位的閱讀!希望分享的內(nèi)容對大家有幫助,更多相關(guān)知識,歡迎關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道!