• Java中的null


    1. null是Java中的关键字,像public、static、final。它是大小写敏感的,你不能将null写成Null或NULL,编译器将不能识别它们然后报错。
    Object obj = NULL; // Not Ok
    Object obj1 = null  //Ok
    
    1. 就像每种原始类型都有默认值一样,如int默认值为0,boolean的默认值为false,null是任何引用类型的默认值,不严格的说是所有object类型的默认值。就像你创建了一个布尔类型的变量,它将false作为自己的默认值,Java中的任何引用变量都将null作为默认值。这对所有变量都是适用的,如成员变量、局部变量、实例变量、静态变量(但当你使用一个没有初始化的局部变量,编译器会警告你)。

    2. 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强制转换成任何引用类型都是可行的,在运行时期都不会抛出空指针异常。

    1. null可以赋值给引用变量,你不能将null赋给基本类型变量,例如int、double、float、boolean。如果你那样做了,编译器将会报错,如下所示:
    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
    

    正如你看到的那样,当你直接将null赋值给基本类型,会出现编译错误。但是如果将null赋值给包装类object,然后将object赋给各自的基本类型,编译器不会报,但是你将会在运行时期遇到空指针异常。这是Java中的自动拆箱导致的。

    1. 任何含有null值的包装类在Java拆箱生成基本数据类型时候都会抛出一个空指针异常。一些程序员犯这样的错误,他们认为自动装箱会将null转换成各自基本类型的默认值,例如对于int转换成0,布尔类型转换成false,但是那是不正确的,如下面所示:
    Integer iAmNull = null;
    int i = iAmNull; // Remember - No Compilation Error
    

    但是当你运行上面的代码片段的时候,你会在控制台上看到主线程抛出空指针异常。在使用HashMap和Integer键值的时候会发生很多这样的错误。当你运行下面代码的时候就会出现错误。

    import java.util.HashMap;
    import java.util.Map;
    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
          }       
        }
    }
    

    输出:

    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:-)

    1. 如果使用了带有null值的引用类型变量,instanceof操作将会返回false:
    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");
    }
    

    输出:

    AmNull is NOT an instance of Integer
    

    这是instanceof操作一个很重要的特性,使得对类型强制转换检查很有用

    1. 不能调用非静态方法来使用一个值为null的引用类型变量,它将会抛出空指针异常。但是可以使用静态方法来使用一个值为null的引用类型变量。因为静态方法使用静态绑定,不会抛出空指针异常。下面是一个例子:
    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");
       }
    

    输出:

    I am static method, can be called by null reference
    Exception in thread "main" java.lang.NullPointerException
                   at Testing.main(Testing.java:11)
    
    1. 你可以将null传递给方法使用,这时方法可以接收任何引用类型,例如public void print(Object obj)可以这样调用print(null)。从编译角度来看这是可以的,但结果完全取决于方法。Null安全的方法,如在这个例子中的print方法,不会抛出空指针异常,只是优雅的退出。如果业务逻辑允许的话,推荐使用null安全的方法。

    2. 你可以使用或者!=操作来比较null值,但是不能使用其他算法或者逻辑操作,例如小于或者大于。跟SQL不一样,在Java中nullnull将返回true,如下所示:

    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){
           }
        }
    }
    

    输出:

    null == null is true in Java
    

    这是关于Java中null的全部。通过Java编程的一些经验和使用简单的技巧来避免空指针异常,你可以使你的代码变得null安全。因为null经常作为空或者未初始化的值,它是困惑的源头。对于方法而言,记录下null作为参数时方法有什么样的行为也是非常重要的。总而言之,记住,null是任何一个引用类型变量的默认值,在java中你不能使用null引用来调用任何的instance方法或者instance变量。

    出处: http://www.importnew.com/14229.html

  • 相关阅读:
    windows anaconda下安装Python的tesserocr库
    windows10上安装docker与碰到的坑
    阿里云centos下部署python flask应用。
    LeetCode--Python合并两个有序链表
    Linux(CentOS)下重置MySQL根(Root)密码,以及远程登录mysql连接IP受限问题解决
    windows下anaconda安装词云wordcloud
    关于selenium使用中谷歌浏览器驱动chromedriver的问题
    LeetCode 184. Department Highest Salary(找出每个部门中最高薪水)
    机器学习七--回归--多元线性回归Multiple Linear Regression
    机器学习六--回归--简单线性回归Simple Linear Regression
  • 原文地址:https://www.cnblogs.com/cag2050/p/7490839.html
Copyright © 2020-2023  润新知