Java數(shù)據(jù)溢出代碼詳解
java是一門相對安全的語言,那么數(shù)據(jù)溢出時它是如何處理的呢? 看一段代碼,
public class Overflow { /** * @param args */ public static void main(String[] args) { int big = 0x7fffffff; //max int value System.out.println("big = " + big); int bigger = big * 4; System.out.println("bigger = " + bigger); } }
運行的結(jié)果是:
big = 2147483647 bigger = -4
int類型在java中是四字節(jié),且分正負,所以最大的int型數(shù)值是0x7fffffff。變量bigger顯然是溢出了,但是你并沒有得到編譯錯誤,或者運行時錯誤??雌饋聿诲e,但是有時候并不一定是好事情,就像這個示例,我們期望的結(jié)果應(yīng)該不是-4,而且程序也沒有錯誤提示。
那么如何解決溢出的問題呢? 很容易想到的一個方案是用更大的數(shù)據(jù)類型,long型占8個字節(jié),可以用,
public class Overflow { /** * @param args */ public static void main(String[] args) { long reallyBig = 0x7fffffffL; System.out.println("reallyBig = " + reallyBig); long reallyBigger = reallyBig * 4; System.out.println("reallyBigger = " + reallyBigger); } }
輸出,
reallyBig = 2147483647 reallyBigger = 8589934588
繼續(xù)思考一個問題,第一個示例溢出的結(jié)果為什么是-4? 可以從第二個示例的結(jié)果得到一些啟示,8589934588的十六進制是1FFFFFFFC,只留四個字節(jié)(int型只占四字節(jié))是FFFFFFFC,這是補碼,并且因為是最高位是1,所以還是個負數(shù)。轉(zhuǎn)成原碼(除符號位外,取反加1)變?yōu)?0000004,也就是-4。
下面一個實例,是從另一篇文章中看到的,也給大家分享下。
long now = System.currentTimeMillis(); long m1 = now-1000*60*60*24*30*3; long test = (long)1000*60*60*24*30*3; long nocast = 1000L*60*60*24*30*3; long m2 = now-1000L*60*60*24*30*3; System.out.println("我是test:"+test); System.out.println(nocast); System.out.println("now:"+now); System.out.println("m1:"+m1); System.out.println("m2:"+m2);
輸出結(jié)果:
我是test:7776000000 7776000000 now:1359510417546 m1:1360324352138 m2:1351734417546
有點奇怪了吧 1000*60*60*24*30*3的計算值與1000L*60*60*24*30*3 的計算值是不一樣的 1000*60*60*24*30*3 的計算結(jié)果是-813934592
而 (long)1000*60*60*24*30*3; 1000L*60*60*24*30*3;的計算結(jié)果都是7776000000 這個結(jié)果才是需要的東西
為什么會出現(xiàn)這種情況呢 Java在做乘法時,如果沒有顯示確定的數(shù)據(jù)類型,那么他會默認按照int數(shù)據(jù)類型進行計算就好比1000*60*60*24*30*3,這個數(shù)值在計算時導(dǎo)致int數(shù)據(jù)溢出,所以出現(xiàn)了負數(shù)的情況。
(long)1000*60*60*24*30*3; 1000L*60*60*24*30*3;這兩者,前者是將首個計算數(shù)強制造型為long 然后按照long類型運算得出了正確的計算結(jié)果,后者在計算時將數(shù)據(jù)類型修改為long 能直接獲得正確的計算結(jié)果,異曲同工。
這一點還是很隱晦的,要注意,防止以后出錯。這錯誤估計不好排查。
還有一點就是 L所在的位置,有點奇怪 1000L*60*60*24*30*3; 1000*60L*60*24*30*3;1000*60*60L*24*30*3;1000*60*60*24*30L*3;這幾種寫法都可以正確轉(zhuǎn)為long型,但是1000*60*60*24*30*3L;就不對,而且3*60*60*24*30*1000這一種寫法的時候 L跟在任何一個數(shù)字后面都沒問題。有點莫名其妙,用的時候debug測試要優(yōu)先確定這個串的計算結(jié)果。
總結(jié)
以上就是本文關(guān)于Java數(shù)據(jù)溢出代碼詳解的全部內(nèi)容,希望對大家有所幫助。歡迎各位瀏覽本站的其他專題,有問題請留言,小編會及時回復(fù)大家的。
- Java編程常見內(nèi)存溢出異常與代碼示例
- Java內(nèi)存溢出和內(nèi)存泄露
- 淺談java內(nèi)存管理與內(nèi)存溢出異常
- 完美解決java讀取大文件內(nèi)存溢出的問題
- Java內(nèi)存區(qū)域與內(nèi)存溢出異常詳解
- Java常見內(nèi)存溢出異常分析與解決
- 了解Java虛擬機JVM的基本結(jié)構(gòu)及JVM的內(nèi)存溢出方式
- 編寫Java代碼制造一個內(nèi)存溢出的情況
- JAVA程序內(nèi)存溢出問題原因分析
- java內(nèi)存溢出示例(堆溢出、棧溢出)
- 基于Java內(nèi)存溢出的解決方法詳解
- java虛擬機內(nèi)存溢出及泄漏實例
相關(guān)文章
Java實現(xiàn)駝峰與下劃線互轉(zhuǎn)的方法
這篇文章主要為大家詳細介紹了Java實現(xiàn)駝峰與下劃線互轉(zhuǎn)的方法,文中的示例代碼講解詳細,感興趣的小伙伴可以跟隨小編一起學習一下2024-04-04mybatis創(chuàng)建項目報Invalid?bound?statement?(not?found)錯誤解決方法
使用MyBatis能夠幫助我們將SQL語句和Java代碼分離,這篇文章主要給大家介紹了關(guān)于mybatis創(chuàng)建項目報Invalid?bound?statement?(not?found)錯誤的解決方法,文中通過實例代碼介紹的非常詳細,需要的朋友可以參考下2023-05-05Java獲取Prometheus監(jiān)控數(shù)據(jù)的方法實現(xiàn)
本文主要介紹了Java獲取Prometheus監(jiān)控數(shù)據(jù)的方法實現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2023-12-12Spring?Cloud?Alibaba?Nacos服務(wù)治理平臺服務(wù)注冊、RestTemplate實現(xiàn)微服務(wù)之間訪
這篇文章主要介紹了Spring?Cloud?Alibaba:Nacos服務(wù)治理平臺,服務(wù)注冊、RestTemplate實現(xiàn)微服務(wù)之間訪問,負載均衡訪問,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2022-06-06