java null 可以转型吗?

Python021

java null 可以转型吗?,第1张

正如我说过的那样,null是Java中一个很重要的概念。null设计初衷是为了表示一些缺失的东西,例如缺失的用户、资源或其他东西。但是,一年后,令人头疼的空指针异常给Java程序员带来不少的骚扰。在这份材料中,我们将学习到Java中null关键字的基本细节,并且探索一些技术来尽可能的减少null的检查以及如何避免恶心的空指针异常。\x0d\x0a1)首先,null是Java中的关键字,像public、static、final。它是大小写敏感的,你不能将null写成Null或NULL,编译器将不能识别它们然后报错。\x0d\x0a\x0d\x0a1\x0d\x0a2\x0d\x0a \x0d\x0aObject obj = NULL// Not Ok\x0d\x0aObject obj1 = null //Ok\x0d\x0a\x0d\x0a使用其他语言的程序员可能会有这个问题,但是现在IDE的使用已经使得这个问题变得微不足道。现在,当你敲代码的时候,IDE像Eclipse、Netbeans可以纠正这个错误。但是使用其他工具像notepad、Vim、Emacs,这个问题却会浪费你宝贵时间的。\x0d\x0a2)就像每种原始类型都有默认值一样,如int默认值为0,boolean的默认值为false,null是任何引用类型的默认值,不严格的说是所有object类型的默认值。就像你创建了一个布尔类型的变量,它将false作为自己的默认值,Java中的任何引用变量都将null作为默认值。这对所有变量都是适用的,如成员变量、局部变量、实例变量、静态变量(但当你使用一个没有初始化的局部变量,编译器会警告你)。为了证明这个事实,你可以通过创建一个变量然后打印它的值来观察这个引用变量,如下图代码所示:\x0d\x0a\x0d\x0aprivate static Object myObj\x0d\x0apublic static void main(String args[]){\x0d\x0aSystem.out.println("What is value of myObjc : " + myObj)\x0d\x0a}\x0d\x0a\x0d\x0a1\x0d\x0a \x0d\x0aWhat is value of myObjc : null\x0d\x0a\x0d\x0a这对静态和非静态的object来说都是正确的。就像你在这里看到的这样,我将myObj定义为静态引用,所以我可以在主方法里直接使用它。注意主方法是静态方法,不可使用非静态变量。\x0d\x0a3)我们要澄清一些误解,null既不是对象也不是一种类型,它仅是一种特殊的值,你可以将其赋予任何引用类型,你也可以将null转化成任何类型,来看下面的代码:\x0d\x0a\x0d\x0aString str = null// null can be assigned to String\x0d\x0aInteger itr = null// you can assign null to Integer also\x0d\x0aDouble dbl = null // null can also be assigned to Double\x0d\x0a \x0d\x0aString myStr = (String) null// null can be type cast to String\x0d\x0aInteger myItr = (Integer) null// it can also be type casted to Integer\x0d\x0aDouble myDbl = (Double) null// yes it's possible, no error\x0d\x0a\x0d\x0a你可以看到在编译和运行时期,将null强制转换成任何引用类型都是可行的,在运行时期都不会抛出空指针异常。\x0d\x0a4)null可以赋值给引用变量,你不能将null赋给基本类型变量,例如int、double、float、boolean。如果你那样做了,编译器将会报错,如下所示:\x0d\x0a\x0d\x0aint i = null// type mismatch : cannot convert from null to int\x0d\x0ashort s = null// type mismatch : cannot convert from null to short\x0d\x0abyte b = null: // type mismatch : cannot convert from null to byte\x0d\x0adouble d = null//type mismatch : cannot convert from null to double\x0d\x0a \x0d\x0aInteger itr = null// this is ok\x0d\x0aint j = itr// this is also ok, but NullPointerException at runtime\x0d\x0a\x0d\x0a正如你看到的那样,当你直接将null赋值给基本类型,会出现编译错误。但是如果将null赋值给包装类object,然后将object赋给各自的基本类型,编译器不会报,但是你将会在运行时期遇到空指针异常。这是Java中的自动拆箱导致的,我们将在下一个要点看到它。\x0d\x0a5) 任何含有null值的包装类在Java拆箱生成基本数据类型时候都会抛出一个空指针异常。一些程序员犯这样的错误,他们认为自动装箱会将null转换成各自基本类型的默认值,例如对于int转换成0,布尔类型转换成false,但是那是不正确的,如下面所示:\x0d\x0a\x0d\x0a1\x0d\x0a2\x0d\x0a \x0d\x0aInteger iAmNull = null\x0d\x0aint i = iAmNull// Remember - No Compilation Error\x0d\x0a\x0d\x0a但是当你运行上面的代码片段的时候,你会在控制台上看到主线程抛出空指针异常。在使用HashMap和Integer键值的时候会发生很多这样的错误。当你运行下面代码的时候就会出现错误。\x0d\x0a\x0d\x0aimport java.util.HashMap\x0d\x0aimport java.util.Map\x0d\x0a \x0d\x0a/**\x0d\x0a * An example of Autoboxing and NullPointerExcpetion\x0d\x0a *\x0d\x0a * @author WINDOWS 8\x0d\x0a */\x0d\x0apublic class Test {\x0d\x0apublic static void main(String args[]) throws InterruptedException {\x0d\x0a Map numberAndCount = new HashMap()\x0d\x0a int[] numbers = {3, 5, 7,9, 11, 13, 17, 19, 2, 3, 5, 33, 12, 5}\x0d\x0a \x0d\x0a for(int i : numbers){\x0d\x0a int count = numberAndCount.get(i)\x0d\x0a numberAndCount.put(i, count++)// NullPointerException here\x0d\x0a } \x0d\x0a}\x0d\x0a}\x0d\x0a\x0d\x0a输出:\x0d\x0a\x0d\x0a1\x0d\x0a2\x0d\x0a \x0d\x0aException in thread "main" java.lang.NullPointerException\x0d\x0a at Test.main(Test.java:25)\x0d\x0a\x0d\x0a这段代码看起来非常简单并且没有错误。你所做的一切是找到一个数字在数组中出现了多少次,这是Java数组中典型的寻找重复的技术。开发者首先得到以前的数值,然后再加一,最后把值放回Map里。程序员可能会以为,调用put方法时,自动装箱会自己处理好将int装箱成Interger,但是他忘记了当一个数字没有计数值的时候,HashMap的get()方法将会返回null,而不是0,因为Integer的默认值是null而不是0。当把null值传递给一个int型变量的时候自动装箱将会返回空指针异常。设想一下,如果这段代码在一个if嵌套里,没有在QA环境下运行,但是你一旦放在生产环境里,BOOM:-)\x0d\x0a6)如果使用了带有null值的引用类型变量,instanceof操作将会返回false:\x0d\x0a\x0d\x0a7\x0d\x0a \x0d\x0aInteger iAmNull = null\x0d\x0aif(iAmNull instanceof Integer){\x0d\x0a System.out.println("iAmNull is instance of Integer") \x0d\x0a \x0d\x0a}else{\x0d\x0a System.out.println("iAmNull is NOT an instance of Integer")\x0d\x0a}\x0d\x0a\x0d\x0a输出:\x0d\x0a\x0d\x0a1\x0d\x0a \x0d\x0ai\x0d\x0a\x0d\x0a1\x0d\x0a \x0d\x0aAmNull is NOT an instance of Integer\x0d\x0a\x0d\x0a这是instanceof操作一个很重要的特性,使得对类型强制转换检查很有用\x0d\x0a7)你可能知道不能调用非静态方法来使用一个值为null的引用类型变量。它将会抛出空指针异常,但是你可能不知道,你可以使用静态方法来使用一个值为null的引用类型变量。因为静态方法使用静态绑定,不会抛出空指针异常。下面是一个例子:\x0d\x0a\x0d\x0a1\x0d\x0a\x0d\x0apublic class Testing {\x0d\x0a public static void main(String args[]){\x0d\x0a Testing myObject = null\x0d\x0a myObject.iAmStaticMethod()\x0d\x0a myObject.iAmNonStaticMethod() \x0d\x0a }\x0d\x0a \x0d\x0a private static void iAmStaticMethod(){\x0d\x0aSystem.out.println("I am static method, can be called by null reference")\x0d\x0a }\x0d\x0a \x0d\x0a private void iAmNonStaticMethod(){\x0d\x0a System.out.println("I am NON static method, don't date to call me by null")\x0d\x0a }\x0d\x0a\x0d\x0a输出:\x0d\x0a\x0d\x0a1\x0d\x0a2\x0d\x0a3\x0d\x0a \x0d\x0aI am static method, can be called by null reference\x0d\x0aException in thread "main" java.lang.NullPointerException\x0d\x0a at Testing.main(Testing.java:11)\x0d\x0a\x0d\x0a8)你可以将null传递给方法使用,这时方法可以接收任何引用类型,例如public void print(Object obj)可以这样调用print(null)。从编译角度来看这是可以的,但结果完全取决于方法。Null安全的方法,如在这个例子中的print方法,不会抛出空指针异常,只是优雅的退出。如果业务逻辑允许的话,推荐使用null安全的方法。\x0d\x0a9)你可以使用==或者!=操作来比较null值,但是不能使用其他算法或者逻辑操作,例如小于或者大于。跟SQL不一样,在Java中null==null将返回true,如下所示:\x0d\x0a\x0d\x0apublic class Test {\x0d\x0a \x0d\x0apublic static void main(String args[]) throws InterruptedException {\x0d\x0a \x0d\x0a String abc = null\x0d\x0a String cde = null\x0d\x0a \x0d\x0a if(abc == cde){\x0d\x0a System.out.println("null == null is true in Java")\x0d\x0a }\x0d\x0a \x0d\x0a if(null != null){\x0d\x0a System.out.println("null != null is false in Java")\x0d\x0a }\x0d\x0a \x0d\x0a // classical null check\x0d\x0a if(abc == null){\x0d\x0a // do something\x0d\x0a }\x0d\x0a \x0d\x0a // not ok, compile time error\x0d\x0a if(abc >null){\x0d\x0a \x0d\x0a }\x0d\x0a}\x0d\x0a}\x0d\x0a\x0d\x0a输出:\x0d\x0a\x0d\x0a1\x0d\x0a \x0d\x0anull == null is true in Java

null不是关键字,是字面常量。

The

Java

Language

Specification中原文:

while

null

might

appear

to

be

a

keyword,

it

is

technically

the

null

literal。

null看起来好像是个关键字,但从技术上讲是个字面常量。