摘要:但是,三目運算符也是有一定的語言規范的。一三目運算符對于條件表達式,先計算條件,然后進行判斷。那么,這段代碼為什么會自動拆箱呢這其實是三目運算符的語法規范。所以,結果就是由于使用了三目運算符,并且第二第三位操作數分別是基本類型和對象。
三目運算符是我們經常在代碼中使用的,a= (b==null?0:1);這樣一行代碼可以代替一個if-else,可以使代碼變得清爽易讀。
但是,三目運算符也是有一定的語言規范的。在運用不恰當的時候會導致意想不到的問題。本文就介紹一個我自己曾經踩過的坑。
一、三目運算符
對于條件表達式b?x:y,先計算條件b,然后進行判斷。如果b的值為true,計算x的值,運算結果為x的值;否則,計算y的值,運算結果為y的值。一個條件表達式從不會既計算x,又計算y。條件運算符是右結合的,也就是說,從右向左分組計算。例如,a?b:c?d:e將按a?b:(c?d:e)執行。
二、自動裝箱與自動拆箱
基本數據類型的自動裝箱(autoboxing)、拆箱(unboxing)是自J2SE 5.0開始提供的功能。
一般我們要創建一個類的對象實例的時候,我們會這樣: Class a = new Class(parameters); 當我們創建一個Integer對象時,卻可以這樣: Integer i = 100;(注意:和 int i = 100;是有區別的 )
實際上,執行上面那句代碼的時候,系統為我們執行了: Integer i = Integer.valueOf(100); 這里暫且不討論這個原理是怎么實現的(何時拆箱、何時裝箱),也略過普通數據類型和對象類型的區別。
我們可以理解為,當我們自己寫的代碼符合裝(拆)箱規范的時候,編譯器就會自動幫我們拆(裝)箱。那么,這種不被程序員控制的自動拆(裝)箱會不會存在什么問題呢?
三、問題回顧
首先,通過你已有的經驗看一下下面這段代碼。如果你得到的結果和后文分析的結果一致(并且你知道原理),那么請忽略本文。如果不一致,請跟我探索下去。
public static void main(String[] args) { Mapmap = new HashMap<>(); Boolean b = map != null ? map.get("test") : false; System.out.println(b); }
以上這段代碼,是我們在不注意的情況下有可能經常會寫的一類代碼(在很多時候我們都愛使用三目運算符)。
一般情況下,我們會認為以上代碼Boolean b的最終得到的值應該是null。因為map.get("test")的值是null,而b又是一個對象,所以得到結果會是null。
但是,以上代碼會拋出NPE:
Exception in thread "main" java.lang.NullPointerException
首先可以明確的是,既然報了空指針,那么一定是有些地方調用了一個null的對象的某些方法。在這短短的兩行代碼中,看上去只有一處方法調用map.get("test"),但是我們也都是知道,map已經事先初始化過了,不會是Null,那么到底是哪里有空指針呢。
我們接下來[反編譯]一下該代碼。看看我們寫的代碼在經過編譯器處理之后變成了什么樣。反編譯后代碼如下:
public static void main(String args[]){ Map map = new HashMap(); Boolean b = Boolean.valueOf(map == null ? false : ((Boolean)map.get("test")).booleanValue()); System.out.println(b); }
看完這段反編譯之后的代碼之后,經過分析我們大概可以知道問題出在哪里。((Boolean)hashmap.get("test")).booleanValue() 的執行過程及結果如下:
hashmap.get("test")->null; (Boolean)null->null; null.booleanValue()->報錯
好,問題終于定位到了。很明顯,上面源代碼中的map.get("test")在被編譯成了
(Boolean)map.get("test").booleanValue(),這是一種自動拆箱的操作。
那么,為什么這里會發生自動拆箱呢?這個問題又如何解決呢?
四、原理分析
通過查看反編譯之后的代碼,我們準確的定位到了問題,分析之后我們可以得出這樣的結論:NPE的原因應該是三目運算符和自動拆箱導致了空指針異常。
那么,這段代碼為什么會自動拆箱呢?這其實是三目運算符的語法規范。參見jls-15.25,摘要如下:
If the second and third operands have the same type (which may be the null type), then that is the type of the conditional expression.
If one of the second and third operands is of primitive type T, and the type of the other is the result of applying boxing conversion (§5.1.7) to T, then the type of the conditional expression is T.
If one of the second and third operands is of the null type and the type of the other is a reference type, then the type of the conditional expression is that reference type.
簡單的來說就是:當第二,第三位操作數分別為基本類型和對象時,其中的對象就會拆箱為基本類型進行操作。
所以,結果就是:由于使用了三目運算符,并且第二、第三位操作數分別是基本類型和對象。所以對對象進行拆箱操作,由于該對象為null,所以在拆箱過程中調用null.booleanValue()的時候就報了NPE。
五、問題解決
如果代碼這么寫,就不會報錯:
Mapmap = new HashMap (); Boolean b = (map!=null ? map.get("test") : Boolean.FALSE);
就是保證了三目運算符的第二第三位操作數都為對象類型。這樣就不會發生自動拆箱操作,以上代碼得到的b的結果為null。
PS:本文中的示例,只是為了更加方便讀者理解三目運算符會導致自動拆箱現象,可能在代碼中并不會直接這樣使用。但是,我自己的代碼確實發生過類似問題。這里簡化一下,為了講清楚原理。
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://specialneedsforspecialkids.com/yun/69350.html
摘要:如果條件為,則邏輯非運算符將得到進行邏輯判斷的順序為從左到右。再次檢測布爾表達式。其作用域限定在循環語句塊,其值與此時數組元素的值相等。 運算符和邏輯控制 運算符 java中的運算符可以分為以下幾種: 算符運算符 關系運算符 位運算符 邏輯運算符 賦值運算符 三目運算符 算符運算符 操作符 描述 + 加法 - 減法 * 乘法 / 除法 % 取余(模) ...
摘要:每日算法題目將碼位于之間的個字符顯示在屏幕上,為了美觀,要求小于的碼值前填充一個,每打印個字符后換行。本文已轉載個人技術公眾號歡迎留言討論與點贊上一篇推薦貓說每日算法實現特殊累加值下一篇推薦貓說每日算法枚舉求就業率問題 Java每日算法 題目 將ASCII碼位于32-126之間的95個字符顯示在屏幕上,為了美觀,要求小于100的碼值前填充一個0,每打印8個字符后換行。 分析 這里先注意...
摘要:不要使用類函數終于,你不用再看到建議不要使用函數的提示了。因為從核心上完全移除了它們,這意味著請你移步至更好的類函數,或者更靈活的層。將從數據庫獲取一個元數據,如果您正在循環訪問特定文章的元數據,則可以在循環中使用它。 showImg(https://segmentfault.com/img/bV75FM?w=1024&h=534); 1. 不要使用 mysql_ 類函數 終于,你不用...
摘要:強制類型轉換下標運算符變量與常量常量是在程序中的不會變化的數據變量其實就是內存中的一個存儲空間,用于存儲數據。表示結束本次循環,繼續下次循環。 Java知識點總結 (基本語法) @(Java知識點總結)[Java, Java基本語法] @(Java開發)[Java基本語法] [toc] Java特點 簡單自然平臺可移植性支持函數式編程JIT 編譯更好的并發編程健壯安全 執行方式 編譯...
摘要:注意,三目運算符中和是成對出現的,最起碼數量上,有幾個就會有幾個 三目運算符相信大家都很熟悉了: foo ? foo == true : foo == false 而三目運算符?:?:?.....的調用方式大家也不陌生, 就相當于一堆if - else if語句: foo ? foo == true : bar ? bar == true : bar == false 但是在zep...
閱讀 1973·2023-04-25 15:45
閱讀 1208·2021-09-29 09:34
閱讀 2500·2021-09-03 10:30
閱讀 2004·2019-08-30 15:56
閱讀 1463·2019-08-29 15:31
閱讀 1271·2019-08-29 15:29
閱讀 3202·2019-08-29 11:24
閱讀 3053·2019-08-26 13:45