Java中有关Null的9件事

对于Java程序员来讲,null是使人头痛的东西。时常会受到空指针异常(NPE)的骚扰。连Java的发明者都认可这是他的一项巨大失误。Java为何要保留null呢?null出现有一段时间了,而且我认为Java发明者知道null与它解决的问题相比带来了更多的麻烦,可是null仍然陪伴着Java。 我愈加感到惊奇,由于java的设计原理是为了简化事情,那就是为何没有浪费时间在指针、操做符重载、多继承实现的缘由,null却与此正好相反。好吧,我真的不知道这个问题的答案,我知道的是无论null被Java开发者和开源社区如何批评,咱们必须与null共同存在。与其为null的存在感到后悔,咱们倒不如更好的学习null,确保正确使用null。 为何在Java中须要学习null?由于若是你对null不注意,Java将使你遭受空指针异常的痛苦,而且你也会获得一个沉痛的教训。精力充沛的编程是一门艺术,你的团队、客户和用户将会更加欣赏你。以个人经验来看,致使空指针异常的一个最主要的缘由是对Java中null的知识还不够。大家当中的不少已经对null很熟悉了,可是对那些不是很熟悉的来讲,能够学到一些关于null老的和新的知识。让咱们一块儿从新学习Java中null的一些重要知识吧。 Java中的Null是什么? 正如我说过的那样,null是Java中一个很重要的概念。null设计初衷是为了表示一些缺失的东西,例如缺失的用户、资源或其余东西。可是,一年后,使人头疼的空指针异常给Java程序员带来很多的骚扰。在这份材料中,咱们将学习到Java中null关键字的基本细节,而且探索一些技术来尽量的减小null的检查以及如何避免恶心的空指针异常。 1)首先,null是Java中的关键字,像public、static、final。它是大小写敏感的,你不能将null写成Null或NULL,编译器将不能识别它们而后报错。java

Object obj = NULL; // Not Ok Object obj1 = null //Ok 使用其余语言的程序员可能会有这个问题,可是如今IDE的使用已经使得这个问题变得微不足道。如今,当你敲代码的时候,IDE像Eclipse、Netbeans能够纠正这个错误。可是使用其余工具像notepad、Vim、Emacs,这个问题却会浪费你宝贵时间的。 2)就像每种原始类型都有默认值同样,如int默认值为0,boolean的默认值为false,null是任何引用类型的默认值,不严格的说是全部object类型的默认值。就像你建立了一个布尔类型的变量,它将false做为本身的默认值,Java中的任何引用变量都将null做为默认值。这对全部变量都是适用的,如成员变量、局部变量、实例变量、静态变量(但当你使用一个没有初始化的局部变量,编译器会警告你)。为了证实这个事实,你能够经过建立一个变量而后打印它的值来观察这个引用变量,以下图代码所示: private static Object myObj; public static void main(String args[]){ System.out.println("What is value of myObjc : " + myObj); } What is value of myObjc : null 这对静态和非静态的object来讲都是正确的。就像你在这里看到的这样,我将myObj定义为静态引用,因此我能够在主方法里直接使用它。注意主方法是静态方法,不可以使用非静态变量。 3)咱们要澄清一些误解,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强制转换成任何引用类型都是可行的,在运行时期都不会抛出空指针异常。 4)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中的自动拆箱致使的,咱们将在下一个要点看到它。 5) 任何含有null值的包装类在Java拆箱生成基本数据类型时候都会抛出一个空指针异常。一些程序员犯这样的错误,他们认为自动装箱会将null转换成各自基本类型的默认值,例如对于int转换成0,布尔类型转换成false,可是那是不正确的,以下面所示:算法

Integer iAmNull = null; int i = iAmNull; // Remember - No Compilation Error 可是当你运行上面的代码片断的时候,你会在控制台上看到主线程抛出空指针异常。在使用HashMap和Integer键值的时候会发生不少这样的错误。当你运行下面代码的时候就会出现错误。sql

import java.util.HashMap; import java.util.Map; /**编程

  • An example of Autoboxing and NullPointerExcpetion
  • @author WINDOWS 8 */ 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:-) 6)若是使用了带有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"); } 输出:数组

i AmNull is NOT an instance of Integer 这是instanceof操做一个很重要的特性,使得对类型强制转换检查颇有用 7)你可能知道不能调用非静态方法来使用一个值为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) 8)你能够将null传递给方法使用,这时方法能够接收任何引用类型,例如public void print(Object obj)能够这样调用print(null)。从编译角度来看这是能够的,但结果彻底取决于方法。Null安全的方法,如在这个例子中的print方法,不会抛出空指针异常,只是优雅的退出。若是业务逻辑容许的话,推荐使用null安全的方法。 9)你可使用==或者!=操做来比较null值,可是不能使用其余算法或者逻辑操做,例如小于或者大于。跟SQL不同,在Java中null==null将返回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变量。 欢迎工做一到五年的Java工程师朋友们加入Java架构开发: 854393687 群内提供免费的Java架构学习资料(里面有高可用、高并发、高性能及分布式、Jvm性能调优、Spring源码,MyBatis,Netty,Redis,Kafka,Mysql,Zookeeper,Tomcat,Docker,Dubbo,Nginx等多个知识点的架构资料)合理利用本身每一分每一秒的时间来学习提高本身,不要再用"没有时间“来掩饰本身思想上的懒惰!趁年轻,使劲拼,给将来的本身一个交代! Java中有关Null的9件事

相关文章
相关标签/搜索