详解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在运行时会自动给变量赋值。赋值原则是整数类型intbyteshortlong的自动赋值为0,带小数点的floatdouble自动赋值为0.0boolean的自动赋值为false,其他各供引用类型变量自动赋值为null

这个具体可以通过调试来看。

四、容器类型与null

List:允许重复元素,可以加入任意多个null

Set:不允许重复元素,最多可以加入一个null

MapMapkey最多可以加入一个nullvalue字段没有限制。

数组:基本类型数组,定义后,如果不给定初始值,则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不一样,Javanull==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

 

 

参考文献:

深入Java关键字null

Java中有关Null9件事

 

原文地址:https://www.cnblogs.com/huangwenjie/p/5655258.html