Java中有關Null的9件事

對于Java程序員來說,null是令人頭痛的東西铝阐。時常會受到空指針異常(NPE)的騷擾址貌。連Java的發(fā)明者都承認這是他的一項巨大失誤。Java為什么要保留null呢徘键?null出現有一段時間了练对,并且我認為Java發(fā)明者知道null與它解決的問題相比帶來了更多的麻煩,但是null仍然陪伴著Java啊鸭。

我越發(fā)感到驚奇锹淌,因為java的設計原理是為了簡化事情,那就是為什么沒有浪費時間在指針赠制、操作符重載赂摆、多繼承實現的原因,null卻與此正好相反钟些。好吧烟号,我真的不知道這個問題的答案,我知道的是不管null被Java開發(fā)者和開源社區(qū)如何批評政恍,我們必須與null共同存在汪拥。與其為null的存在感到后悔,我們倒不如更好的學習null篙耗,確保正確使用null迫筑。

為什么在Java中需要學習null?因為如果你對null不注意宗弯,Java將使你遭受空指針異常的痛苦脯燃,并且你也會得到一個沉痛的教訓萝风。精力充沛的編程是一門藝術诱篷,你的團隊昌渤、客戶和用戶將會更加欣賞你屉符。以我的經驗來看,導致空指針異常的一個最主要的原因是對Java中null的知識還不夠重父。你們當中的很多已經對null很熟悉了僻弹,但是對那些不是很熟悉的來說炉菲,可以學到一些關于null老的和新的知識详恼。讓我們一起重新學習Java中null的一些重要知識吧补君。

Java中的Null是什么?

正如我說過的那樣昧互,null是Java中一個很重要的概念赚哗。null設計初衷是為了表示一些缺失的東西她紫,例如缺失的用戶、資源或其他東西屿储。但是,一年后渐逃,令人頭疼的空指針異常給Java程序員帶來不少的騷擾够掠。在這份材料中,我們將學習到Java中null關鍵字的基本細節(jié)茄菊,并且探索一些技術來盡可能的減少null的檢查以及如何避免惡心的空指針異常疯潭。

1)首先,null是Java中的關鍵字面殖,像public竖哩、static、final脊僚。它是大小寫敏感的相叁,你不能將null寫成Null或NULL,編譯器將不能識別它們然后報錯辽幌。

1
2
Object obj = NULL; // Not Ok
Object obj1 = null //Ok
使用其他語言的程序員可能會有這個問題增淹,但是現在IDE的使用已經使得這個問題變得微不足道。現在乌企,當你敲代碼的時候虑润,IDE像Eclipse、Netbeans可以糾正這個錯誤加酵。但是使用其他工具像notepad拳喻、Vim、Emacs猪腕,這個問題卻會浪費你寶貴時間的冗澈。
2)就像每種原始類型都有默認值一樣,如int默認值為0码撰,boolean的默認值為false渗柿,null是任何引用類型的默認值,不嚴格的說是所有object類型的默認值脖岛。就像你創(chuàng)建了一個布爾類型的變量朵栖,它將false作為自己的默認值,Java中的任何引用變量都將null作為默認值柴梆。這對所有變量都是適用的陨溅,如成員變量、局部變量绍在、實例變量门扇、靜態(tài)變量(但當你使用一個沒有初始化的局部變量雹有,編譯器會警告你)。為了證明這個事實臼寄,你可以通過創(chuàng)建一個變量然后打印它的值來觀察這個引用變量霸奕,如下圖代碼所示:

1
2
3
4
private static Object myObj;
public static void main(String args[]){
System.out.println("What is value of myObjc : " + myObj);
}
1
What is value of myObjc : null
這對靜態(tài)和非靜態(tài)的object來說都是正確的。就像你在這里看到的這樣吉拳,我將myObj定義為靜態(tài)引用质帅,所以我可以在主方法里直接使用它。注意主方法是靜態(tài)方法留攒,不可使用非靜態(tài)變量煤惩。

3)我們要澄清一些誤解,null既不是對象也不是一種類型炼邀,它僅是一種特殊的值魄揉,你可以將其賦予任何引用類型,你也可以將null轉化成任何類型拭宁,來看下面的代碼:

1
2
3
4
5
6
7
String str = null; // null can be assigned to String
Integer itr = null; // you can assign null to Integer also
Double dbl = null; // null can also be assigned to Double

String myStr = (String) null; // null can be type cast to String
Integer myItr = (Integer) null; // it can also be type casted to Integer
Double myDbl = (Double) null; // yes it's possible, no error
你可以看到在編譯和運行時期洛退,將null強制轉換成任何引用類型都是可行的,在運行時期都不會拋出空指針異常红淡。
4)null可以賦值給引用變量不狮,你不能將null賦給基本類型變量,例如int在旱、double摇零、float、boolean桶蝎。如果你那樣做了驻仅,編譯器將會報錯,如下所示:

1
2
3
4
5
6
7
int i = null; // type mismatch : cannot convert from null to int
short s = null; // type mismatch : cannot convert from null to short
byte b = null: // type mismatch : cannot convert from null to byte
double d = null; //type mismatch : cannot convert from null to double

Integer itr = null; // this is ok
int j = itr; // this is also ok, but NullPointerException at runtime
正如你看到的那樣登渣,當你直接將null賦值給基本類型噪服,會出現編譯錯誤。但是如果將null賦值給包裝類object胜茧,然后將object賦給各自的基本類型粘优,編譯器不會報,但是你將會在運行時期遇到空指針異常呻顽。這是Java中的自動拆箱導致的雹顺,我們將在下一個要點看到它。
5) 任何含有null值的包裝類在Java拆箱生成基本數據類型時候都會拋出一個空指針異常廊遍。一些程序員犯這樣的錯誤嬉愧,他們認為自動裝箱會將null轉換成各自基本類型的默認值,例如對于int轉換成0喉前,布爾類型轉換成false没酣,但是那是不正確的王财,如下面所示:

1
2
Integer iAmNull = null;
int i = iAmNull; // Remember - No Compilation Error
但是當你運行上面的代碼片段的時候,你會在控制臺上看到主線程拋出空指針異常裕便。在使用HashMap和Integer鍵值的時候會發(fā)生很多這樣的錯誤绒净。當你運行下面代碼的時候就會出現錯誤。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
import java.util.HashMap;
import java.util.Map;

/**

  • An example of Autoboxing and NullPointerExcpetion

  • @author WINDOWS 8
    */
    public class Test {
    public static void main(String args[]) throws InterruptedException {
    Map numberAndCount = new HashMap<>();
    int[] numbers = {3, 5, 7,9, 11, 13, 17, 19, 2, 3, 5, 33, 12, 5};

    for(int i : numbers){
    int count = numberAndCount.get(i);
    numberAndCount.put(i, count++); // NullPointerException here
    }
    }
    }
    輸出:

1
2
Exception in thread "main" java.lang.NullPointerException
at Test.main(Test.java:25)
這段代碼看起來非常簡單并且沒有錯誤闪金。你所做的一切是找到一個數字在數組中出現了多少次疯溺,這是Java數組中典型的尋找重復的技術。開發(fā)者首先得到以前的數值哎垦,然后再加一,最后把值放回Map里恃疯。程序員可能會以為漏设,調用put方法時,自動裝箱會自己處理好將int裝箱成Interger今妄,但是他忘記了當一個數字沒有計數值的時候郑口,HashMap的get()方法將會返回null,而不是0盾鳞,因為Integer的默認值是null而不是0犬性。當把null值傳遞給一個int型變量的時候自動裝箱將會返回空指針異常。設想一下腾仅,如果這段代碼在一個if嵌套里乒裆,沒有在QA環(huán)境下運行,但是你一旦放在生產環(huán)境里推励,BOOM:-)

6)如果使用了帶有null值的引用類型變量鹤耍,instanceof操作將會返回false:

1
2
3
4
5
6
7
Integer iAmNull = null;
if(iAmNull instanceof Integer){
System.out.println("iAmNull is instance of Integer");

}else{
System.out.println("iAmNull is NOT an instance of Integer");
}
輸出:

1
i
1
AmNull is NOT an instance of Integer
這是instanceof操作一個很重要的特性,使得對類型強制轉換檢查很有用

7)你可能知道不能調用非靜態(tài)方法來使用一個值為null的引用類型變量验辞。它將會拋出空指針異常稿黄,但是你可能不知道,你可以使用靜態(tài)方法來使用一個值為null的引用類型變量跌造。因為靜態(tài)方法使用靜態(tài)綁定杆怕,不會拋出空指針異常。下面是一個例子:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
public class Testing {
public static void main(String args[]){
Testing myObject = null;
myObject.iAmStaticMethod();
myObject.iAmNonStaticMethod();
}

private static void iAmStaticMethod(){
System.out.println("I am static method, can be called by null reference");
}

private void iAmNonStaticMethod(){
System.out.println("I am NON static method, don't date to call me by null");
}
輸出:

1
2
3
I am static method, can be called by null reference
Exception in thread "main" java.lang.NullPointerException
at Testing.main(Testing.java:11)
8)你可以將null傳遞給方法使用壳贪,這時方法可以接收任何引用類型陵珍,例如public void print(Object obj)可以這樣調用print(null)。從編譯角度來看這是可以的撑碴,但結果完全取決于方法撑教。Null安全的方法,如在這個例子中的print方法醉拓,不會拋出空指針異常伟姐,只是優(yōu)雅的退出收苏。如果業(yè)務邏輯允許的話,推薦使用null安全的方法愤兵。

9)你可以使用==或者!=操作來比較null值鹿霸,但是不能使用其他算法或者邏輯操作,例如小于或者大于秆乳。跟SQL不一樣懦鼠,在Java中null==null將返回true,如下所示:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
public class Test {

public static void main(String args[]) throws InterruptedException {

   String abc = null;
   String cde = null;

   if(abc == cde){
       System.out.println("null == null is true in Java");
   }

   if(null != null){
       System.out.println("null != null is false in Java"); 
   }

   // classical null check
   if(abc == null){
       // do something
   }

   // not ok, compile time error
   if(abc > null){

   }
}

}
輸出:

1
null == null is true in Java
這是關于Java中null的全部屹堰。通過Java編程的一些經驗和使用簡單的技巧來避免空指針異常肛冶,你可以使你的代碼變得null安全。因為null經常作為空或者未初始化的值扯键,它是困惑的源頭睦袖。對于方法而言,記錄下null作為參數時方法有什么樣的行為也是非常重要的荣刑∠隗希總而言之,記住厉亏,null是任何一個引用類型變量的默認值董习,在java中你不能使用null引用來調用任何的instance方法或者instance變量。

**轉載自:
原文鏈接: javarevisited 翻譯: ImportNew.com - Calarence
譯文鏈接: http://www.importnew.com/14229.html

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
  • 序言:七十年代末爱只,一起剝皮案震驚了整個濱河市皿淋,隨后出現的幾起案子,更是在濱河造成了極大的恐慌虱颗,老刑警劉巖沥匈,帶你破解...
    沈念sama閱讀 218,204評論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現場離奇詭異忘渔,居然都是意外死亡高帖,警方通過查閱死者的電腦和手機,發(fā)現死者居然都...
    沈念sama閱讀 93,091評論 3 395
  • 文/潘曉璐 我一進店門畦粮,熙熙樓的掌柜王于貴愁眉苦臉地迎上來散址,“玉大人,你說我怎么就攤上這事宣赔≡铮” “怎么了?”我有些...
    開封第一講書人閱讀 164,548評論 0 354
  • 文/不壞的土叔 我叫張陵儒将,是天一觀的道長吏祸。 經常有香客問我,道長钩蚊,這世上最難降的妖魔是什么贡翘? 我笑而不...
    開封第一講書人閱讀 58,657評論 1 293
  • 正文 為了忘掉前任蹈矮,我火速辦了婚禮,結果婚禮上鸣驱,老公的妹妹穿的比我還像新娘泛鸟。我一直安慰自己,他們只是感情好踊东,可當我...
    茶點故事閱讀 67,689評論 6 392
  • 文/花漫 我一把揭開白布北滥。 她就那樣靜靜地躺著,像睡著了一般闸翅。 火紅的嫁衣襯著肌膚如雪再芋。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,554評論 1 305
  • 那天坚冀,我揣著相機與錄音祝闻,去河邊找鬼。 笑死遗菠,一個胖子當著我的面吹牛,可吹牛的內容都是我干的华蜒。 我是一名探鬼主播辙纬,決...
    沈念sama閱讀 40,302評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼叭喜!你這毒婦竟也來了贺拣?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 39,216評論 0 276
  • 序言:老撾萬榮一對情侶失蹤捂蕴,失蹤者是張志新(化名)和其女友劉穎譬涡,沒想到半個月后,有當地人在樹林里發(fā)現了一具尸體啥辨,經...
    沈念sama閱讀 45,661評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡涡匀,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 37,851評論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現自己被綠了溉知。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片陨瘩。...
    茶點故事閱讀 39,977評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖级乍,靈堂內的尸體忽然破棺而出舌劳,到底是詐尸還是另有隱情,我是刑警寧澤玫荣,帶...
    沈念sama閱讀 35,697評論 5 347
  • 正文 年R本政府宣布甚淡,位于F島的核電站,受9級特大地震影響捅厂,放射性物質發(fā)生泄漏贯卦。R本人自食惡果不足惜资柔,卻給世界環(huán)境...
    茶點故事閱讀 41,306評論 3 330
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望脸侥。 院中可真熱鬧建邓,春花似錦、人聲如沸睁枕。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,898評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽外遇。三九已至注簿,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間跳仿,已是汗流浹背诡渴。 一陣腳步聲響...
    開封第一講書人閱讀 33,019評論 1 270
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留菲语,地道東北人妄辩。 一個月前我還...
    沈念sama閱讀 48,138評論 3 370
  • 正文 我出身青樓,卻偏偏與公主長得像山上,于是被迫代替她去往敵國和親眼耀。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 44,927評論 2 355