您好,登錄后才能下訂單哦!
這篇文章將為大家詳細講解有關Java三目運算中隱藏自動拆裝箱的示例分析,小編覺得挺實用的,因此分享給大家做個參考,希望大家閱讀完這篇文章后可以有所收獲。
最近修改線上bug的時候排查了一個十分隱藏的bug,直接上代碼:
Integer a = null; boolean flag = true; Integer b = flag ? a : 0;
乍一看是沒什么毛病的,但是已運行就會發現報空指針,在idea里面也會警告可能有空指針,這是什么原因呢?
直接看字節碼:
0: aconst_null 1: astore_1 2: iconst_1 3: istore_2 4: iload_2 5: ifeq 15 8: aload_1 9: invokevirtual #2 // Method java/lang/Integer.intValue:()I 12: goto 16 15: iconst_0 16: invokestatic #3 // Method java/lang/Integer.valueOf:(I)Ljava/lang/Integer; 19: astore_3 20: getstatic #4 // Field java/lang/System.out:Ljava/io/PrintStream; 23: aload_3 24: invokevirtual #5 // Method java/io/PrintStream.println:(Ljava/lang/Object;)V 27: return
可以看到字節碼中調用了`Integer.valueOf()`
方法,因為我們代碼中一個值使用的是0(基本數據類型int),編譯器就會進行自動拆裝箱(成int),
雖然三目運算的后面邏輯不會執行,但是隱藏的自動拆裝箱會執行`Integer.valueOf()`方法,也就有了空指針異常。
為了進一步驗證存在自動拆裝箱,把代碼修改一下:
Integer a = null; boolean flag = true; Integer b = flag ? a : new Integer(0);
再看字節碼:
0: aconst_null 1: astore_1 2: iconst_1 3: istore_2 4: iload_2 5: ifeq 12 8: aload_1 9: goto 20 12: new #2 // class java/lang/Integer 15: dup 16: iconst_0 17: invokespecial #3 // Method java/lang/Integer."<init>":(I)V 20: astore_3 21: getstatic #4 // Field java/lang/System.out:Ljava/io/PrintStream; 24: aload_3 25: invokevirtual #5 // Method java/io/PrintStream.println:(Ljava/lang/Object;)V
可以看到,由于重新創建了一個`Integer`對象,并沒有基本類型的存在,也就不存在自動拆裝箱,修改過后的代碼也就不會有問題了,但是idea的警告依舊存在。
這是一個非常隱蔽,也非常容易忽略和踩坑的一個地方,三目運算符的使用應該保證后面的值都是常量,或者統一類型,不然就會出現上面的情況。
更甚三目運算符本身提供的作用也不過是為了簡化邏輯,在其中放入過多的邏輯判斷也就違背了其初衷。
關于“Java三目運算中隱藏自動拆裝箱的示例分析”這篇文章就分享到這里了,希望以上內容可以對大家有一定的幫助,使各位可以學到更多知識,如果覺得文章不錯,請把它分享出去讓更多的人看到。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。