正如我说过的那样,null是Java中一个很重要的概念。null设计初衷是为了表示一些缺失的东西,例如缺失的用户、资源或其他东西。但是,一年后,令人头疼的空指针异常给Java程序员带来不少的骚扰。在这份材料中,我们将学习到Java中null关键字的基本细节,并且探索一些技术来尽可能的减少null的检查以及如何避免恶心的空指针异常。
?
1)首先,null是Java中的关键字,像public、static、final。它是大小写敏感的,你不能将null写成Null或NULL,编译器将不能识别它们然后报错。
1
2
|
Object obj = NULL; // Not Ok
Object obj1 = null ? //Ok
|
2)就像每种原始类型都有默认值一样,如int默认值为0,boolean的默认值为false,null是任何引用类型的默认值,不严格的说是所有object类型的默认值。就像你创建了一个布尔类型的变量,它将false作为自己的默认值,Java中的任何引用变量都将null作为默认值。这对所有变量都是适用的,如成员变量、局部变量、实例变量、静态变量(但当你使用一个没有初始化的局部变量,编译器会警告你)。为了证明这个事实,你可以通过创建一个变量然后打印它的值来观察这个引用变量,如下图代码所示:
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 |
这对静态和非静态的object来说都是正确的。就像你在这里看到的这样,我将myObj定义为静态引用,所以我可以在主方法里直接使用它。注意主方法是静态方法,不可使用非静态变量。
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
|
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
|
5) 任何含有null值的包装类在Java拆箱生成基本数据类型时候都会抛出一个空指针异常。一些程序员犯这样的错误,他们认为自动装箱会将null转换成各自基本类型的默认值,例如对于int转换成0,布尔类型转换成false,但是那是不正确的,如下面所示:
1
2
|
Integer iAmNull = null ;
int i = iAmNull; // Remember - No Compilation Error
|
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数组中典型的寻找重复的技术。开发者首先得到以前的数值,然后再加一,最后把值放回Map里。程序员可能会以为,调用put方法时,自动装箱会自己处理好将int装箱成Interger,但是他忘记了当一个数字没有计数值的时候,HashMap的get()方法将会返回null,而不是0,因为Integer的默认值是null而不是0。当把null值传递给一个int型变量的时候自动装箱将会返回空指针异常。设想一下,如果这段代码在一个if嵌套里,没有在QA环境下运行,但是你一旦放在生产环境里,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)你可能知道不能调用非静态方法来使用一个值为null的引用类型变量。它将会抛出空指针异常,但是你可能不知道,你可以使用静态方法来使用一个值为null的引用类型变量。因为静态方法使用静态绑定,不会抛出空指针异常。下面是一个例子:
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方法,不会抛出空指针异常,只是优雅的退出。如果业务逻辑允许的话,推荐使用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变量。
原文:http://tomcat-oracle.iteye.com/blog/2172223