国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

這么簡單的三目運算符竟然也有這么多坑?

開發 前端
最近,阿粉在一個業務改造中,使用三目運算符重構了業務代碼,沒想到測試的時候竟然發生 NPE 的問題。

[[328294]]

最近,阿粉在一個業務改造中,使用三目運算符重構了業務代碼,沒想到測試的時候竟然發生 NPE 的問題。

[[328295]]

重構代碼非常簡單,代碼如下:

  1. // 方法返回參數類型為 Integer 
  2. //  private Integer code; 
  3. SimpleObj simpleObj = new SimpleObj(); 
  4. // 其他業務邏輯 
  5. if (simpleObj == null) { 
  6.     return -1; 
  7. else { 
  8.     return simpleObj.getCode(); 

這段 if 判斷,阿粉看到的時候,感覺很是繁瑣,于是使用三目運算符重構了一把,代碼如下:

  1. // 方法返回參數類型為 Integer 
  2. SimpleObj simpleObj = new SimpleObj(); 
  3. // 其他業務邏輯 
  4. return simpleObj == null ? -1 : simpleObj.getCode(); 

測試的時候,第四行代碼拋出了空指針,這里代碼很簡單,顯然只有 simpleObj#getCode才有可能發生 NPE 問題。

但是我明明為 simpleObj做過判空判斷,simpleObj 對象肯定不是 null,那么只有 simpleObj#getCode 返回為 null。但是我的代碼并沒有對這個方法返回值做任何操作,為何會觸發 NPE?

難道是又是自動拆箱導致的 NPE 問題?

在解答這個問題之前,我們首先復習一下三目運算符。

三目運算符

三目運算符,官方英文名稱:Conditional Operator ? :,中文直譯條件表達式,本文不糾結名稱,統一使用三目運算符。

三目運算符的基本用法非常簡單,它由三個操作數的運算符構成,形式為:

  1. <表達式 1><表達式 2>:<表達式 3> 

三目運算符的計算從左往右計算,首先需要計算計算表達式 1 ,其結果類型必須為 Boolean 或 boolean,否則發生編譯錯誤。

當表達式 1 的結果為 true,將會執行表達式 2,否則將會執行表達式 3。

表達式 2 與表達式 3 最后的類型必須得有返回結果,即不能為是 void,若為 void ,編譯時將會報錯。

最后需要注意的是,表達式 2 與表達式 3 不會被同時執行,兩者只有一個會被執行。

踩坑案例

了解完三目運算符的基本原理,我們簡化一下開頭例子,復現一下三目運算符使用過程的一些坑。假設我們的例子簡化成如下:

  1. boolean flag = true; //設置成true,保證表達式 2 被執行 
  2. int simpleInt = 66; 
  3. Integer nullInteger = null

案例 1

第一個案例我們根據如下計算 result 的值。

  1. int result = flag ? nullInteger : simpleInt; 

這個案例為開頭的例子的簡化版本,運算上述代碼,將會發生 NPE 的。

為什么會發發生 NPE 呢?

這里可以給大家一個小技巧,當我們從代碼上沒辦法找到答案時,我們可以試試查看一下編譯之后字節碼,或許是 Java 編譯之后增加某些東西,從而導致問題。

使用 javap -s -c class 查看 class 文件字節碼,如下:

可以看到字節碼中加入一個拆箱操作,而這個拆箱只有可能發生在 nullInteger。

那么為什么 Java 編譯器在編譯時會對表達式進行拆箱?難道所有數字類型的包裝類型都會進行拆箱嗎?

三目運算符表達式發生自動拆箱,其實官方在 「The Java Language Specification(簡稱:JLS)」15.25 節[1]中做出一些規定,部分內容如下:

  • JDK7 規范
  • 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.

用大白話講,如果表達式 2 與表達式 3 類型相同,那么這個不用任何轉換,三目運算符表達式結果當然與表達式 2,3 類型一致。

當表達 2 或表達式 3 其中任一一個是基本數據類型,比如 int,而另一個表達式類型為包裝類型,比如 Integer,那么三目運算符表達式結果類型將會為基本數據類型,即 int。

  • ps:有沒有疑問?為什么不規定最后結果類型都為包裝類那?

這是 Java 語言層面一種規范,但是這個規范如果強制讓程序員執行,想必平常使用三目運算符將會比較麻煩。所以面對這種情況, Java 在編譯器在編譯過程加入自動拆箱進制。

所以上述代碼可以等同于下述代碼:

  1. int result = flag ? nullInteger.intValue() : simpleInt; 

如果我們一開始的代碼如上所示,那么這里錯誤點其實就很明顯了。

案例 2

接下來我們在第一個案例基礎上修改一下:

  1. boolean flag = true; //設置成true,保證表達式 2 被執行 
  2. int simpleInt = 66; 
  3. Integer nullInteger = null
  4. Integer objInteger = Integer.valueOf(88); 
  5.  
  6. int result = flag ? nullInteger : objInteger; 

運行上述代碼,依然會發生 NPE 的問題。當然這次問題發生點與上一個案例不一樣,但是錯誤原因卻是一樣,還是因為自動拆箱機制導致。

這一次表達式 2 與表達式 3 都為包裝類 Integer,所以三目運算符的最后結果類型也會是 Integer。

但是由于 result是 int 基本數據類型,好家伙,數據類型不一致,編譯器將會對三目運算符的結果進行自動拆箱。由于結果為 null,自動拆箱將報錯了。

上述代碼等同為:

  1. int result = (flag ? nullInteger : objInteger).intValue(); 

案例 3

我們再稍微改造一下案例 1 的例子,如下所示:

  1. boolean flag = true; //設置成true,保證表達式 2 被執行 
  2. int simpleInt = 66; 
  3. Integer nullInteger = null
  4. Integer result = flag ? nullInteger : simpleInt; 

案例 3 與案例 1 右邊部分完全相同,只不過左邊部分的類型不一樣,一個為基本數據類型 int,一個為 Integer。

按照案例 1 的分析,這個也會發生 NPE 問題,原因與案例 1 一樣。

這個之所以拿出來,其實想說下,上述三目運算符的結果為 int 類型,而左邊類型為 Integer,所以這里將會發生自動裝箱操作,將 int類型轉化為 Integer。

上述代碼等同為:

  1. Integer result = Integer.valueOf(flag ? nullInteger.intValue() : simpleInt); 

案例 4

最后一個案例,與上面案例都不一樣,代碼如下:

  1. boolean flag = true; //設置成true,保證表達式 2 被執行 
  2. Integer nullInteger = null
  3. Long objLong = Long.valueOf(88l); 
  4.  
  5. Object result = flag ? nullInteger : objLong; 

運行上述代碼,依然將會發生 NPE 的問題。

這個案例表達式 2 與表達式 3 類型不一樣,一個為 Integer,一個為 Long,但是這兩個類型都是 Number的子類。

面對上述情況,JLS 規定:

  • Otherwise, binary numeric promotion (§5.6.2[2]) is applied to the operand types, and the type of the conditional expression is the promoted type of the second and third operands.
  • Note that binary numeric promotion performs value set conversion (§5.1.13[3]) and may perform unboxing conversion (§5.1.8[4]).

大白話講,當表達式 2 與表達式 3 類型不一致,但是都為數字類型時,低范圍類型將會自動轉為高范圍數據類型,即向上轉型。這個過程將會發生自動拆箱。

  • Java 中向上轉型并不需要添加任何轉化,但是向下轉換必須強制添加類型轉換。

上述代碼轉化比較麻煩,我們先從字節碼上來看:

第一步,將 nullInteger拆箱。

第二步,將上一步的值轉為 long 類型,即 (long)nullInteger.intValue()。

第三步,由于表達式 2 變成了基本數據類型,表達式 3 為包裝類型,根據案例 1 講到的規則,包裝類型需要轉為基本數據類型,所以表達式 3 發生了拆箱。

第四步,由于三目運算符最后的結果類型為基本數據類型:long,但是左邊類型為 Object,這里就需要把 long 類型裝箱轉為包裝類型。

所以最后代碼等同于:

  1. Object result = Long.valueOf(flag ? (long)nullInteger.intValue() : objLong.longValue()); 

總結

看完上述四個案例,想必大家應該會有種感受,沒想到這么簡單的三目運算符,既然暗藏這么多「殺機」。

不過大家也不用過度害怕,不使用三目運算符。只要我們在開發過程重點注意包裝類型的自動拆箱問題就好了,另外也要注意三目運算符的計算結果再賦值的時候自動拆箱引發的 NPE 的問題。

最好大家在開發過程中,都遵守一定的規范,即保持表達式 2 與表達式 3 的類型一致,不讓 Java 編譯器有自動拆箱的機會。

建議大家沒事經常看下阿里出品的『Java 開發手冊』,在最新的「泰山版」就增加三目運算符的這一節規范。

ps:公號消息回復:『開發手冊』,獲取最新版的 Java 開發手冊。

最后一定要做好的單元測試,不要慣性思維,覺得這么簡單的一個東西,看起來根本不可能出錯的。

參考資料

[1]15.25 節: https://docs.oracle.com/javase/specs/jls/se7/html/jls-15.html#jls-15.25

[2]§5.6.2: https://docs.oracle.com/javase/specs/jls/se7/html/jls-5.html#jls-5.6.2

[3]§5.1.13: https://docs.oracle.com/javase/specs/jls/se7/html/jls-5.html#jls-5.1.13

[4]§5.1.8: https://docs.oracle.com/javase/specs/jls/se7/html/jls-5.html#jls-5.1.8

[5]Java 開發手冊》解讀:三目運算符為何會導致 NPE?: https://developer.aliyun.com/article/758784

 

責任編輯:武曉燕 來源: Java極客技術
相關推薦

2022-07-06 11:47:27

JAVAfor循環

2018-06-26 15:00:24

Docker安全風險

2024-04-02 08:41:10

ArrayListSubList場景

2021-02-03 20:19:08

Istio流量網格

2025-07-23 10:13:57

2017-12-21 19:38:50

潤乾中間表

2022-07-26 23:43:29

編程語言開發Java

2021-01-14 05:08:44

編譯鏈接

2013-01-15 09:41:45

編程語言

2023-11-13 08:49:54

2024-02-20 08:09:51

Java 8DateUtilsDate工具類

2017-07-04 14:01:40

機房機柜

2017-07-12 08:20:32

閃存用途企業

2021-06-10 09:00:33

單例模式數據庫

2022-03-03 07:00:43

Mybatiswhere標簽

2018-12-11 12:58:20

CPU散熱器鰭片

2019-08-27 08:17:57

云計算安全云服務商

2017-06-16 16:16:36

庫存扣減查詢

2018-05-29 14:57:59

HashMap容量初始化

2025-05-26 09:30:00

SQL數據庫索引
點贊
收藏

51CTO技術棧公眾號

三上悠亚一区二区| 成人小视频免费在线观看| 精品香蕉一区二区三区| 制服丝袜影音先锋| 337p粉嫩大胆噜噜噜噜噜91av| 欧美一区二区影视| 亚洲承认在线| 国产精品香蕉视屏| 午夜久久99| 99高清视频有精品视频| 成人看的视频| 国产精品亚洲自拍| 欧美女王vk| 人人爽久久涩噜噜噜网站| 噜噜噜天天躁狠狠躁夜夜精品| 九九热这里只有精品免费看| 台湾成人免费视频| 日韩在线观看免费高清| 色豆豆成人网| 美女av一区二区| 精品国产一区二区三区性色av | 我要色综合中文字幕| 色综合伊人色综合网站| 国产精品久久久久77777丨| 亚洲欧美精品在线| 男女啪啪999亚洲精品| 日本美女一区二区三区视频| 亚洲永久免费观看| 久久久久电影| 精品无码久久久久久久动漫| 噜噜噜91成人网| 欧美aaa在线观看| 成人av电影在线| 日本xxxx黄色| 在线影视一区二区三区| 毛片在线播放视频| 国产成人精品亚洲午夜麻豆| 久久久久久久久影视| 捆绑调教美女网站视频一区| 中文字幕一区二区三区乱码| 狠狠久久亚洲欧美| 欧美 日韩 亚洲 一区| 亚洲国产成人在线| 一个人免费观看视频www在线播放 一个人免费视频www在线观看 | 国产成a人无v码亚洲福利| 日韩黄色片在线| 久久久不卡网国产精品二区| 依人在线免费视频| 精品久久久久久| 欧美1—12sexvideos| 久久精彩免费视频| 成人在线电影在线观看视频| 精品国产乱码久久久久久郑州公司| 老司机午夜精品视频| 黄页网站大全在线观看| 一区二区三区精品| 182tv在线播放| 最近2019中文字幕mv免费看| 欧美日韩在线播放三区四区| 男人在线资源站| 国产福利资源一区| 国产裸体写真av一区二区| 国产一区日韩一区| 在线观看精品视频| 亚洲欧洲性图库| 又爽又大又黄a级毛片在线视频| 精品视频在线播放| 亚洲香蕉视频| 日韩一区二区电影在线观看| 久久免费精品国产久精品久久久久| 亚洲国产精华液| 精品少妇一区二区三区日产乱码 | 91精品入口蜜桃| 国产黑丝在线一区二区三区| 妞干网在线视频| 91精品麻豆日日躁夜夜躁| 粉嫩av一区二区三区四区五区 | 中文字幕日韩一区| gogogogo高清视频在线| 欧美大片大片在线播放| 亚洲精品极品| 成人影院在线观看视频| 欧美一区二区免费视频| 亚洲免费成人av在线| 国产欧美自拍视频| 精品久久久久久久久久| 成人黄色在线| 免费成人在线观看av| 亚洲欧美日韩国产另类专区| 涩涩视频在线播放| 电影午夜精品一区二区三区| 久久久久国产一区二区三区四区| 999国产在线视频| 81精品国产乱码久久久久久| 久草中文综合在线| 成人精品福利| 国产精品久久久久不卡| 成人免费观看男女羞羞视频| 成人高清免费在线播放| 欧美在线一级va免费观看| 国产成人在线电影| 成人日韩欧美| http;//www.99re视频| 国产精品久久久久久久久果冻传媒 | 在线观看国产精品日韩av| 一区二区三区午夜视频| 欧美 日韩 国产一区| 欧美一级黄色大片| 四虎成人av| 成人黄色电影在线| 欧美日产国产成人免费图片| 国产乱对白刺激视频不卡| 黄色免费在线观看| 亚洲直播在线一区| 亚洲国产人成综合网站| 国产色噜噜噜91在线精品| 日本人体一区二区| 亚洲欧美国产一本综合首页| 三级久久三级久久| 黄色国产网站在线播放| 99在线国产| 色女孩综合影院| 亚洲成人最新网站| 最新在线地址| 国产精品主播视频| 久久久成人av毛片免费观看| 中文字幕黄色大片| 亚洲精品av在线播放| 久久最新视频| 黄网站在线免费| 蜜桃av久久久亚洲精品| 欧美日韩情趣电影| 亚洲精品社区| 99热国产在线| 日韩黄色影视| 亚洲国产黄色片| 国产精选一区二区三区| 午夜欧美巨大性欧美巨大 | 欧美日韩视频免费在线观看| 国产网站在线播放| 欧美丰满少妇xxxx| 国产欧美日韩三级| 欧美另类中文字幕| 亚洲成人av免费看| 97久久久免费福利网址| 亚洲视频网在线直播| 精品久久影院| 日本韩国精品一区二区| 国产午夜精品在线| 亚洲精品在线免费观看视频| 99精品国产在热久久| 七七成人影院| 成人免费网站入口| 久久成人在线视频| 中文字幕亚洲区| 成人av动漫在线观看| 欧美18xxxxx| 蜜桃av久久久亚洲精品| 亚洲人成在线播放| 国产日本亚洲高清| 午夜国产精品视频| av福利在线导航| 人妻有码中文字幕| 国产精品三级在线| 欧美一区二区三区视频| 国产一区二三区| 日本欧美高清| 成人资源www网在线最新版| 国产又爽又黄ai换脸| 久久免费观看视频| 一区二区视频| 免费av一级电影| 欧美成熟毛茸茸复古| 中文字幕日韩欧美在线| 亚洲一区二区欧美激情| 免费在线成人网| www.豆豆成人网.com| 黄色网址在线播放| 国产精品久久久久7777| 日本一区二区三区四区视频| 欧美日韩成人在线| 久久精品欧美一区二区三区麻豆| 国产高清一区二区| 日韩a**中文字幕| 最近中文字幕mv2018在线高清| 欧美人与性禽动交精品| 久久久久久久久久久人体| 欧美三级欧美一级| 久久久99免费| 久久精品观看| 国产不卡一区| 欧亚一区二区| 超碰在线国产| 欧美黄色性生活| 亚洲 国产 欧美一区| 国产成人av网址| 正在播放国产一区| 欧美午夜精品一区二区三区| 久久精子c满五个校花| 天堂影院一区二区|