摘要:還有需要注意的一點是,此類緩存行為不僅存在于對象。還存在于其他的整數類型,,,。但是能改變緩存范圍的就只有了。
前言
最近跟許多朋友聊了下,在這個“跳槽”的黃金季節,大家都有點蠢蠢欲動,所以最近就多聊聊面試的時候需要注意的一些問題,這些問題不一定多深奧,多復雜,但是一不注意的話卻容易掉坑。下面看一下面試的時候經常遇到的一段代碼:
public class IntegerDemo { public static void main(String[] args) { Integer numA = 127; Integer numB = 127; Integer numC = 128; Integer numD = 128; System.out.println("numA == numB : "+ (numA == numB)); System.out.println("numC == numD : "+ (numC == numD)); } }
根據大家以往的經驗,會認為上面的代碼用“==“符號來比較,對比的是對象的引用,那么ABCD是不同的對象,所以輸出當然是false了。我在《“==”、“equals()”、“hashcode()”之間的秘密》這篇文章也討論過。那么事實也是如此嗎?下面看一下輸出結果:
numA == numB : true numC == numD : false
What?這個輸出結果怎么跟以往的認知有所出入呢?在我們的代碼“Integer numA = 127”中,編譯器會把基本數據的“自動裝箱”(autoboxing)成包裝類,所以這行代碼就等價于“Integer numA = Integer.valueOf(127)”了,這樣我們就可以進入valueOf方法查看它的實現原理。
//Integer valueOf方法 public static Integer valueOf(int i) { if (i >= IntegerCache.low && i <= IntegerCache.high) return IntegerCache.cache[i + (-IntegerCache.low)]; return new Integer(i); } //Integer靜態內部類 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() {} }
從上面的源碼可以看到,valueOf方法會先判斷傳進來的參數是否在IntegerCache的low與high之間,如果是的話就返回cache數組里面的緩存值,不是的話就new Integer(i)返回。
那我們再往上看一下IntegerCache,它是Integer的內部靜態類,low默認是-128,high的值默認127,但是high可以通過JVM啟動參數XX:AutoBoxCacheMax=size來修改(如圖),如果我們按照這樣修改了,然后再次執行上面代碼,這時候2次輸出都是true,因為緩存的區間變成-128~200了。
但是如果我們是通過構造器來生成一個Integer對象的話,下面的輸出都是false。因為這樣不會走ValueOf方法,所以按照正常的對象對比邏輯即可。
public class IntegerDemo { public static void main(String[] args) { Integer numA = new Integer(127); Integer numB = new Integer(127); Integer numC = new Integer(128); Integer numD = new Integer(128); System.out.println("numA == numB : "+ (numA == numB));//false System.out.println("numC == numD : "+ (numC == numD));//false } }
還有需要注意的一點是,此類緩存行為不僅存在于Integer對象。還存在于其他的整數類型Byte,Short,Long,Character。但是能改變緩存范圍的就只有Integer了。
結語有時候往往越簡單的知識越容易掉坑里,所以要保持自己的求知欲,不斷鞏固的基礎,才能讓自己在面試的時候不會栽跟頭。
文章始發于微信公眾號《深夜里的程序猿》,每天分享最干的干貨,轉載請注明出處,侵權必究。
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://specialneedsforspecialkids.com/yun/77509.html
摘要:清楚了以上流程,我們直接來看函數主要用作初始化應用監聽端口以及啟動。其中就是保存聊天室所有聊天消息的結構。關于的解讀我會放到閱讀源碼時講。然后把消息加到緩存里,如果緩存大于限制則取最新的條消息。 tornado 源碼自帶了豐富的 demo ,這篇文章主要分析 demo 中的聊天室應用: chatdemo 首先看 chatdemo 的目錄結構: ├── chatdemo.py ├── ...
摘要:類實際上是中中的緩存類,目的是節省內存消耗,提高程序性能。而當堆內存中的對象存儲非常多時,就有可能造成內存泄漏。使用頻率高創建對象也就越多,堆內存中的對象也就越多,所以也就會可能發生上述中的內存溢出等問題。 面試題:問以下代碼輸出的結果是多少? public class IntegerTest { @Test public void test() { ...
閱讀 3556·2023-04-25 16:35
閱讀 686·2021-10-11 11:09
閱讀 6137·2021-09-22 15:11
閱讀 3352·2019-08-30 14:03
閱讀 2590·2019-08-29 16:54
閱讀 3343·2019-08-29 16:34
閱讀 3042·2019-08-29 12:18
閱讀 2113·2019-08-28 18:31