详解null
前言
在java中初始化的时候经常用到null,也经常会碰到空指针异常(NullPointerException),由于碰到的频率比较高,我认为有必要进行一下了解,揭开它的神秘面纱。
一、null是代表不确定的对象
Java中,null是一个关键字,用来标识一个不确定的对象。因此可以将null赋给引用类型变量,但不可以将null赋给基本类型变量。
比如:int a = null;是错误的。Ojbect o = null是正确的。
Java中,变量的适用都遵循一个原则,先定义,并且初始化后,才可以使用。我们不能int a后,不给a指定值,就去打印a的值。这条对对于引用类型变量也是适用的。
有时候,我们定义一个引用类型变量,在刚开始的时候,无法给出一个确定的值,但是不指定值,程序可能会在try语句块中初始化值。这时候,我们下面使用变量的时候就会报错。这时候,可以先给变量指定一个null值,问题就解决了。例如:
Connection conn = null; try { conn = DriverManager.getConnection("url", "user", "password"); } catch (SQLException e) { e.printStackTrace(); }
String catalog = conn.getCatalog(); |
如果刚开始的时候不指定conn = null,则最后一句编译器会认为这个对象没有初始化,就会报错。
二、null本身既不是一种类型,也不是一个对象
null本身虽然能代表一个不确定的对象,但就null本身来说,它不是对象,也不知道什么类型,也不是java.lang.Object的实例。
可以做一个简单的例子:
// null是对象吗? 属于Object类型吗? if (null instanceof java.lang.Object) { System.out.println("null属于java.lang.Object类型"); } else { System.out.println("null不属于java.lang.Object类型"); } |
结果会输出:null不属于java.lang.Object类型。
null不属于java.lang.Object类型 |
有趣的是:
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"); } |
结果输出:
iAmNull is NOT an instance of Integer |
即:当对象初始化为null时,它只是null,并不是这个类的实例。
初始化为null和未初始化的区别:
调用方法时的报错不同,一个是运行时异常,即空指针异常;一个编译时异常。
同时你也可以将null转成任意类型。
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强制转换成任何引用类型都是可行的,在运行时期都不会抛出空指针异常。
但是也没看出来这种强转有任何意义。
三、Java默认给引用类型变量赋值null
在定义变量的时候,如果定义后没有给变量赋值,则Java在运行时会自动给变量赋值。赋值原则是整数类型int、byte、short、long的自动赋值为0,带小数点的float、double自动赋值为0.0,boolean的自动赋值为false,其他各供引用类型变量自动赋值为null。
这个具体可以通过调试来看。
四、容器类型与null
List:允许重复元素,可以加入任意多个null。
Set:不允许重复元素,最多可以加入一个null。
Map:Map的key最多可以加入一个null,value字段没有限制。
数组:基本类型数组,定义后,如果不给定初始值,则java运行时会自动给定值。引用类型数组,不给定初始值,则所有的元素值为null。
五、可以使用静态方法来使用一个值为null的引用类型变量
你可能知道不能调用非静态方法来使用一个值为null的引用类型变量。它将会抛出空指针异常,但是你可能不知道,你可以使用静态方法来使用一个值为null的引用类型变量。因为静态方法使用静态绑定,不会抛出空指针异常。下面是一个例子:
public static void main(String args[]) { Test1 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"); } |
结果输出:
I am static method, can be called by null reference Exception in thread "main" java.lang.NullPointerException at test._null.Test1.main(Test1.java:58) |
六、 可以使用==或者!=操作来比较null值
你可以使用==或者!=操作来比较null值,但是不能使用其他算法或者逻辑操作,例如小于或者大于。跟SQL不一样,在Java中null==null将返回true,如下所示:
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) {
}*/ |
结果输出:
null == null is true in Java |
参考文献: