标签:
在java中初始化的时候经常用到null,也经常会碰到空指针异常(NullPointerException),由于碰到的频率比较高,我认为有必要进行一下了解,揭开它的神秘面纱。
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本身来说,它不是对象,也不知道什么类型,也不是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在运行时会自动给变量赋值。赋值原则是整数类型int、byte、short、long的自动赋值为0,带小数点的float、double自动赋值为0.0,boolean的自动赋值为false,其他各供引用类型变量自动赋值为null。
这个具体可以通过调试来看。
List:允许重复元素,可以加入任意多个null。
Set:不允许重复元素,最多可以加入一个null。
Map:Map的key最多可以加入一个null,value字段没有限制。
数组:基本类型数组,定义后,如果不给定初始值,则java运行时会自动给定值。引用类型数组,不给定初始值,则所有的元素值为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值,但是不能使用其他算法或者逻辑操作,例如小于或者大于。跟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 |
?
?
?
标签:
原文地址:http://www.cnblogs.com/huangwenjie/p/5655258.html