1. 程式人生 > >Java中的三目運算符可能出現的問題

Java中的三目運算符可能出現的問題

-m add 運算符 inline test orm JD borde lean

你真的了解Java中的三目運算符嗎?

2018-04-27 刨根問底的 Hollis Hollis Hollis 技術分享圖片

hollischuang

一個對Coding有著獨特追求的人。

技術分享圖片

技術分享圖片

三目運算符是我們經常在代碼中使用的,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) {
Map<String, Boolean> map = 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。

五、問題解決

如果代碼這麽寫,就不會報錯:

Map<String,Boolean> map =  new HashMap<String, Boolean>();
Boolean b = (map!=null ? map.get("test") : Boolean.FALSE);

就是保證了三目運算符的第二第三位操作數都為對象類型。這樣就不會發生自動拆箱操作,以上代碼得到的b的結果為null。

PS:本文中的示例,只是為了更加方便讀者理解三目運算符會導致自動拆箱現象,可能在代碼中並不會直接這樣使用。但是,我自己的代碼確實發生過類似問題。這裏簡化一下,為了講清楚原理。

Java中的三目運算符可能出現的問題