原文:http://www.importnew.com/14229.html
我们要澄清一些误解,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)
|