程序员人生 网站导航

Java中有关 null 的9件事

栏目:php教程时间:2015-08-13 08:07:27

Java中有关 null 的9件事

对Java程序员来讲,null是使人头痛的东西。经常会遭到空指针异常(NPE)的骚扰。

连Java的发明者都承认这是他的1项巨大失误。Java为何要保存null呢?null出现有1段时间了,并且我认为Java发明者知道null与它解决的问题相比带来了更多的麻烦,但是null依然陪伴着Java。

我越发感到惊奇,由于java的设计原理是为了简化事情,那就是为何没有浪费时间在指针、操作符重载、多继承实现的缘由,null却与此正好相反。好吧,我真的不知道这个问题的答案,我知道的是不管null被Java开发者和开源社区如何批评,我们必须与null共同存在。与其为null的存在感到后悔,我们倒不如更好的学习null,确保正确使用null。

为何在Java中需要学习null?由于如果你对null不注意,Java将使你遭受空指针异常的痛苦,并且你也会得到1个沉痛的教训。精力充分的编程是1门艺术,你的团队、客户和用户将会更加欣赏你。以我的经验来看,致使空指针异常的1个最主要的缘由是对Java中null的知识还不够。你们当中的很多已对null很熟习了,但是对那些不是很熟习的来讲,可以学到1些关于null老的和新的知识。让我们1起重新学习Java中null的1些重要知识吧。


Java中的 null 是甚么 ?

正如我说过的那样,null是Java中1个很重要的概念。null设计初衷是为了表示1些缺失的东西,例如缺失的用户、资源或其他东西。但是,1年后,使人头疼的空指针异常给Java程序员带来很多的骚扰。在这份材料中,我们将学习到Java中null关键字的基本细节,并且探索1些技术来尽量的减少null的检查和如何避免恶心的空指针异常。

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

Object obj = NULL; // Not Ok Object obj1 = null //Ok

使用其他语言的程序员可能会有这个问题,但是现在IDE的使用已使得这个问题变得微不足道。现在,当你敲代码的时候,IDE像Eclipse、Netbeans可以纠正这个毛病。但是使用其他工具像notepad、Vim、Emacs,这个问题却会浪费你宝贵时间的。


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

privatestatic Object myObj; publicstatic void main(String args[]){ System.out.println("What is value of myObjc : " + myObj); }
What is value of myObjc : null

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

3)我们要澄清1些误解,null既不是对象也不是1种类型,它仅是1种特殊的值,你可以将其赋予任何援用类型,你也能够将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强迫转换成任何援用类型都是可行的,在运行时期都不会抛出空指针异常。


4)null可以赋值给援用变量,你不能将null赋给基本类型变量,例如int、double、float、boolean。如果你那样做了,编译器将会报错,以下所示:

inti = null;// type mismatch : cannot convert from null to int shorts = null;// type mismatch : cannot convert from null to short byteb = null:// type mismatch : cannot convert from null to byte doubled = null;//type mismatch : cannot convert from null to double Integer itr = null;// this is ok intj = itr; // this is also ok, but NullPointerException at runtime

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


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

Integer iAmNull = null; inti = iAmNull; // Remember - No Compilation Error

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

importjava.util.HashMap; importjava.util.Map; /** * An example of Autoboxing and NullPointerExcpetion * * @author WINDOWS 8 */ publicclass Test { publicstatic void main(String args[]) throwsInterruptedException { Map numberAndCount = newHashMap<>(); int[] numbers = {3,5,7,9,11,13,17,19,2,3,5,33,12,5}; for(inti : numbers){ intcount = numberAndCount.get(i); numberAndCount.put(i, count++); // NullPointerException here } } }


输出:

Exceptioninthread "main"java.lang.NullPointerException at Test.main(Test.java:25)

这段代码看起来非常简单并且没有毛病。你所做的1切是找到1个数字在数组中出现了多少次,这是Java数组中典型的寻觅重复的技术。开发者首先得到之前的数值,然后再加1,最后把值放回Map里。程序员可能会以为,调用put方法时,自动装箱会自己处理好将int装箱成Interger,但是他忘记了当1个数字没有计数值的时候,HashMap的get()方法将会返回null,而不是0,由于Integer的默许值是null而不是0。当把null值传递给1个int型变量的时候自动装箱将会返回空指针异常。假想1下,如果这段代码在1个if嵌套里,没有在QA环境下运行,但是你1旦放在生产环境里,BOOM:-)


6)如果使用了带有null值的援用类型变量,instanceof操作将会返回false:

Integer iAmNull = null; if(iAmNullinstanceofInteger){ System.out.println("iAmNull is instance of Integer"); }else{ System.out.println("iAmNull is NOT an instance of Integer"); }

输出:

i

AmNull is NOT an instance of Integer

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


7)你可能知道不能调用非静态方法来使用1个值为null的援用类型变量。它将会抛出空指针异常,但是你可能不知道,你可使用静态方法来使用1个值为null的援用类型变量。由于静态方法使用静态绑定,不会抛出空指针异常。下面是1个例子:

publicclass Testing { publicstatic void main(String args[]){ Testing myObject = null; myObject.iAmStaticMethod(); myObject.iAmNonStaticMethod(); } privatestatic void iAmStaticMethod(){ System.out.println("I am static method, can be called by null reference"); } privatevoid 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 Exceptioninthread "main"java.lang.NullPointerException at Testing.main(Testing.java:11)

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

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

publicclass Test { publicstatic void main(String args[]) throwsInterruptedException { 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 truein Java

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


------分隔线----------------------------
------分隔线----------------------------

最新技术推荐