Java中值型別和引用型別的比較與問題解決
一、問題描述
前幾天因為一個需求出現了Bug。說高階點也挺高階,說白點也很簡單。其實也就是一個很簡單的Java基礎入門時候的值型別和引用型別的區別。只是開發的時候由於自己的問題,導致小問題的出現。還好突然想起來以前看過一篇對於該問題講解的部落格,才能快速定位問題的位置。防止下次再犯,順便也就把這個當做筆記記錄下來,放入自己的Bug集中。
二、值型別和引用型別的比較
這個大家應該都是沒問題的,很簡單。值型別比較是比較值,引用型別是比較地址。對於正常的操作來說,比較值型別我們可以直接使用 == ,引用型別就使用equals來做比較就不會出現問題。
引用型別
/** * 測試Integer */ public static void test_Integer(){ Integer number_01 = 10; Integer number_02 = 10; System.out.println(number_01.equals(number_02)); }
上面的測試結果很明顯是true,絕對沒有問題的。
值型別
/** * 測試int */ public static void test_Int(){ int number_01 = 10; int number_02 = 10; System.out.println(number_01 == number_02); }
上面的測試結果很明顯是true,絕對沒有問題的。
三、問題
但是問題就出現在,開發的使用為了防止出現為null的時候會被系統使用0來代替,所以就使用了Integer型別來做操作,並且在比較的時候用了 == 。這就很尷尬了,開始自測完全沒出現問題,因為沒到達記錄數。很開心,把程式碼提交下班,妥妥的。但是尷尬的事情來了,測試報告出現在了郵箱裡面了。
初始沒問題的情況
/** * 測試Integer */ public static void test_Integer(){ Integer number_001 = 10; Integer number_002 = 10; System.out.println(number_001 == number_002); }
結果:
當記錄超過一定數的時候,出現問題
/** * 測試Integer */ public static void test_Integer(){ Integer number_001 = 128; Integer number_002 = 128; System.out.println(number_001 == number_002); }
結果:
四、解決
後面一想,很快確定問題了。是自己的馬虎,偷懶使用了 == ,造成這次問題的出現,當改為equals就可以妥妥的回家了。開始自測沒問題主要還是因為Integer 的快取搞的事情。扒拉到Integer的原始碼,發現裡面用了快取機制,對-128~127的值做了快取,如果在這個值區間內使用==來做比較的話,比較的就是值了,所以才造成開始以為沒問題,後面運行了一段時間後就出現問題了。當不在值區間內就必須使用equals來完成比較。
private static class IntegerCache { static final int low = -128; static final int high; static final Integer cache[]; static { // high value may be configured by property int h = 127; String integerCacheHighPropValue = sun.misc.VM.getSavedProperty("java.lang.Integer.IntegerCache.high"); if (integerCacheHighPropValue != null) { try { int i = parseInt(integerCacheHighPropValue); i = Math.max(i,127); // Maximum array size is Integer.MAX_VALUE h = Math.min(i,Integer.MAX_VALUE - (-low) -1); } catch( NumberFormatException nfe) { // If the property cannot be parsed into an int,ignore it. } } high = h; cache = new Integer[(high - low) + 1]; int j = low; for(int k = 0; k < cache.length; k++) cache[k] = new Integer(j++); // range [-128,127] must be interned (JLS7 5.1.7) assert IntegerCache.high >= 127; } private IntegerCache() {} }
五、總結
當初面試的時候這種東西應該是背的滾瓜爛熟,絕對可以應對面試。但是一旦開發起來就是各種問題都出現了,而且這種東西還不會報出錯日誌,純屬開發問題。歸總來說還是自己的水平不夠,還需要繼續提高。下次要防止這種低階問題的出現,很尷尬。同樣也是對自己學的東西要融匯貫通,而不是每學一個單獨的知識點就夠了,沒有起到聯通的效果。原始碼扒拉一下還是會加深自己的印象。
好了,以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,謝謝大家對我們的支援。