溫馨提示×

溫馨提示×

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

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

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

發(fā)布時(shí)間:2021-08-04 11:05:48 來源:億速云 閱讀:285 作者:小新 欄目:編程語言

這篇文章主要介紹Java中如何解決在時(shí)間戳計(jì)算過程中遇到的數(shù)據(jù)溢出問題,文中介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們一定要看完!

背景

在跑定時(shí)任務(wù)的過程中,發(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ì)算的過程中30 * 24 * 60 * 60 * 1000計(jì)算結(jié)果大于Integer.MAX_VALUE,所以出現(xiàn)了數(shù)據(jù)溢出,從而導(dǎo)致了計(jì)算結(jié)果不準(zhǔn)確的問題。

驗(yà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ù)期不一樣,因?yàn)?0 * 86400000 = 2592000000,但是計(jì)算出來卻是:-1702967296。

到這里想必大家都知道原因了,這是因?yàn)閖ava中整數(shù)的默認(rèn)類型是整型int,而int的最大值是2147483647,

在代碼中java是先計(jì)算右值,再賦值給long變量的。在計(jì)算右值的過程中(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)該就沒有什么問題了,但是這樣就真的保險(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é)果也不對,正確的方式應(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ì)算過程中遇到的數(shù)據(jù)溢出問題”這篇文章的所有內(nèi)容,感謝各位的閱讀!希望分享的內(nèi)容對大家有幫助,更多相關(guān)知識,歡迎關(guān)注億速云行業(yè)資訊頻道!

向AI問一下細(xì)節(jié)

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

AI