NULL

null是Java中一个很重要的概念。null设计初衷是为了表示一些缺失的东西,例如缺失的用户、资源或其他东西。


1)首先,null是Java中的关键字,像public、static、final。它是大小写敏感的,你不能将null写成Null或NULL,编译器将不能识别它们然后报错。

1
2
Object obj = NULL; // false
Object obj1 = null  //true
当你敲代码的时候,IDE像Eclipse、Netbeans可以纠正这个错误。但是使用其他工具像notepad、Vim、Emacs,这个问题却会浪费你宝贵时间的。

2)就像每种原始类型都有默认值一样,如int默认值为0,boolean的默认值为false,null是任何引用类型的默认值,不严格的说是所有object类型的默认值。就像你创建了一个布尔类型的变量,它将false作为自己的默认值,Java中的任何引用变量都将null作为默认值。这对所有变量都是适用的,如成员变量、局部变量实例变量、静态变量(但当你使用一个没有初始化的局部变量,编译器会警告你)。为了证明这个事实,你可以通过创建一个变量然后打印它的值来观察这个引用变量,如下图代码所示:

1
2
3
4
private static Object m;
public static void main(String args[]){
     System.out.println( "What is value of myObjc : " +m);
}
1
What is value of m : null

这对静态和非静态的object来说都是正确的。就像你在这里看到的这样,我将m定义为静态引用,所以我可以在主方法里直接使用它。注意主方法是静态方法,不可使用非静态变量。

3)null既不是对象也不是一种类型,它仅是一种特殊的值,你可以将其赋予任何引用类型,你也可以将null转化成任何类型,来看下面的代码:

1
2
3
4
5
6
7
String str = null ; //可以将null赋值为字符串
Integer itr = null ; //也可以将null赋值为整数
Double dbl = null //空也可以被赋值为Double
 
String myStr = (String) null ; //null可以为字符串类型转换为字符串
Integer myItr = (Integer) null ; //它也可以为整数类型为整数
Double myDbl = (Double) null ; //是的,这是可能的,没有错误
你可以将null强制转换成任何引用类型都是可行的,在运行时期都不会抛出空指针异常。

4)null可以赋值给引用变量,不能将null赋给基本类型变量,例如int、double、float、boolean。如下所示:

1
2
3
4
5
6
7
int i = null ; //类型不匹配:不能从null转换为int
short s = null ; //  类型不匹配:不能从null转换为短
byte b = null : //类型不匹配:不能从null转换为字节
double d = null ; //类型不匹配:不能从null转换为double
 
Integer itr = null ; // true
int j = itr; //这也可以,但是在运行时是NullPointerException
当你直接将null赋值给基本类型,会出现编译错误。但是如果将null赋值给包装类object,然后将object赋给各自的基本类型,编译器不会报,但是你将会在运行时期遇到空指针异常。这是Java中的自动拆箱导致的。

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

1
2
Integer iAmNull = null ;
int i = iAmNull; // true
但是当你运行上面的代码片段的时候,你会在控制台上看到主线程抛出空指针异常。当你运行下面代码的时候就会出现错误。
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;
 
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环境下运行,但是你一旦放在生产环境里.

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( "我是静态方法,可以用空引用来调用" );
    }
 
    private void iAmNonStaticMethod(){
        System.out.println( "我是非静态的方法,不要用空来调用我" );
    }

输出:

1
2
3
我是静态方法,可以用空引用来调用
Exception in thread "main" java.lang.NullPointerException
                at Testing.main(Testing.java:11)

8)你可以将null传递给方法使用,这时方法可以接收任何引用类型,例如public void print(Object obj)可以这样调用print(null)。从编译角度来看这是可以的,但结果完全取决于方法。Null安全的方法,如在这个例子中的print方法,不会抛出空指针异常,只是优雅的退出。如果业务逻辑允许的话,推荐使用null安全的方法。

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

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
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" );
        }
 
        //null检查
        if (abc == null ){
            // do something
        }
 
        //不可以,编译时错误
        if (abc > null ){
 
        }
     }
}

输出:

1
null == null is true in Java

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

10)空指针异常产生的主要原因如下:
(1) 当一个对象不存在时又调用其方法会产生异常 obj.method() // obj对象不存在(2) 当访问或修改一个对象不存在的字段时会产生常 obj.method()  // method方法不存在
 具体情况如下:
空指针错误:使用基本的Java数据类型,变量的值要么已经是默认值,如果没有对其正常赋值,程序便不能通过编译,因此使用基本的Java数据类型(double,float,boolean,char,int,long)一般不会引起空指针异常。由此可见,空指针异常主要跟与对象的操作相关。
情况一:不管对象是否为空就直接开始使用。
  代码段1:
  out.println(request.getParameter("username"));
描述:
   代码段1的功能十分简单,就是输出用户输入的表域"username"的值。
说明:
  看上去,上面的语句找不出什么语法错误,而且在大多数情况下也遇不到什么问题。但是,如果某个用户在输入数据时并没有提供表单域"username" 的值,或通过某种途径绕过表单直接输入时,此时request.getParameter("username")的值为空(不是空字符串,是空对象 null。)。
情况二: 即使对象可能为空时,也调用Java.lang.Object或Object对象本身的一些方法如toString(), equals(Object obj)等操作。
代码段2:
  String userName = request.getParameter("username");
  If (userName.equals("root")) 
  {....}
描述:
  代码段2的功能是检测用户提供的用户名,如果是用户名称为"root"的用户时,就执行一些特别的操作。
说明:
  在代码段2中,如果有用户没有提供表单域"username"的值时,字符串对象userName为null值,不能够将一个null的对象与另一个对象直接比较,同样,代码段2就会抛出(Java.lang.NullPointerException)空指针错误。
代码段3:
          String userName = session.getAttribute
          ("session.username").toString();
描述:
  代码段3的功能是将session中session.username的值取出,并将该值赋给字符串对象 userName。
说明:
  在一般情况下,如果在用户已经进行某个会话,则不会出现什么问题;但是,如果此时应用服务器重新启动,而用户还没有重新登录,(也可能是用户关闭浏览器,但是仍打开原来的页面。)那么,此时该session的值就会失效,同时导致session中的session.username的值为空。对一个为 null的对象的直接执行toString()操作,就会导致系统抛出(Java.lang.NullPointerException)空指针异常。
 
解决方案:


   为了确保进行操作或引用的对象非空,假若我们要对某对象进行操作或引用,我们首先去检查该对象是否已经实例化且不为空;并且在系统中加入针对对象为空时情况的处理。
  如:采用String对象保存用户提交的结果;在如果涉及对象的操作时,先检测其是否为空后,检查到对象为空后,可再选择进行以下任一种处理方式:
  处理方式 1) 检查到对象为空时,设置对象值为空字符串或一个默认值;
  处理方式 2) 检测到对象为空时,根本不执行某操作,直接跳转到其他处理中。
  处理方式 3) 检查到对象为空时,提示用户操作有错误。
  将代码段2按以上方式进行改写,得到:
方式1:
  String userName = request.getParameter("username");
   // 该变量值为空时,转化为默认空字符串
  If (userName == null)
  userName = "";
  If (userName.equals("root"))
  {..........}
方式2:
  String userName = request.getParameter("username");
  // 该变量值为空时,转化为默认空字符串,不执行有关操作。
  If (usreName != null)
  {
  If (userName.equals("root"))
  {..........}
    }
方式3:
  String userName = request.getParameter("username");
  // 该变量值为空时,转化为默认空字符串,不执行有关操作。
  If (usreName == null)
  {  // 提示用户输入信息为空
  }
实际中,上面提供到三种处理方式也同样适用于其他异常的处理:
异常处理方式 1) 检查到异常出现,设置对象值为空字符串或一个默认值;
异常处理方式 2) 检测到异常出现,根本不执行某操作,直接跳转到其他处理中。
异常处理方式 3) 检查到异常出现,提示用户操作有错误。
 


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值