溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

怎么解決Java在時間戳計算的過程中遇到的數(shù)據(jù)溢出問題

發(fā)布時間:2021-11-20 11:31:32 來源:億速云 閱讀:259 作者:iii 欄目:編程語言

本篇內(nèi)容介紹了“怎么解決Java在時間戳計算的過程中遇到的數(shù)據(jù)溢出問題”的有關知識,在實際案例的操作過程中,不少人都會遇到這樣的困境,接下來就讓小編帶領大家學習一下如何處理這些情況吧!希望大家仔細閱讀,能夠學有所成!

背景

今天在跑定時任務的過程中,發(fā)現(xiàn)有一個任務在設置數(shù)據(jù)的查詢時間范圍異常,出現(xiàn)了開始時間戳比結束時間戳大的奇怪現(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);  }}

先放出結論:因為java中整數(shù)默認是int類型,在計算的過程中30 * 24 * 60 * 60 * 1000計算結果大于Integer.MAX_VALUE,所以出現(xiàn)了數(shù)據(jù)溢出,從而導致了計算結果不準確的問題。

驗證

我們將上面的代碼稍稍改造一下,方便我們確認定位問題,調(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);  }} 

結果如下:

end : 1560869539864 minus mills : -1702967296start: 1562572507160

這和我們的預期不一樣,因為30 * 86400000 = 2592000000,但是計算出來卻是:-1702967296。

到這里想必大家都知道原因了,這是因為java中整數(shù)的默認類型是整型int,而int的最大值是2147483647,

在代碼中java是先計算右值,再賦值給long變量的。在計算右值的過程中(int型相乘)發(fā)生溢出,然后將溢出后截斷的值賦給變量,導致了結果不準確。

將代碼做一下小小的改動,再看一下。

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);  }}

結果為

end : 1560869539864 minus mills : 2592000000start: 1558277539864

似乎這樣應該就沒有什么問題了,但是這樣就真的保險了嗎,如果我要把30調(diào)整為24856(Integer.MAX_VALUE / 86400 = 24855),即改為:long a = 24856 * 24 * 60 * 60 * 1000L 那么同樣會出現(xiàn)溢出。

因為java的運算規(guī)則從左到右,再與最后一個long型的1000相乘之前就已經(jī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)容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關的知識可以關注億速云網(wǎng)站,小編將為大家輸出更多高質量的實用文章!

向AI問一下細節(jié)

免責聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權內(nèi)容。

AI