Java 中有关 Null 的 9 件事

本贴最后更新于 2724 天前,其中的信息可能已经水流花落

对于 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,编译器将不能识别它们然后报错。
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 键值的时候会发生很多这样的错误。当你运行下面代码的时候就会出现错误。

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 变量。

原文链接: javarevisited 翻译: ImportNew.com - Calarence
译文链接: http://www.importnew.com/14229.html
[ 转载请保留原文出处、译者和译文链接。]

  • Java

    Java 是一种可以撰写跨平台应用软件的面向对象的程序设计语言,是由 Sun Microsystems 公司于 1995 年 5 月推出的。Java 技术具有卓越的通用性、高效性、平台移植性和安全性。

    3167 引用 • 8207 回帖
  • null
    8 引用 • 37 回帖

相关帖子

欢迎来到这里!

我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。

注册 关于
请输入回帖内容 ...

推荐标签 标签

  • FFmpeg

    FFmpeg 是一套可以用来记录、转换数字音频、视频,并能将其转化为流的开源计算机程序。

    22 引用 • 31 回帖 • 3 关注
  • Hexo

    Hexo 是一款快速、简洁且高效的博客框架,使用 Node.js 编写。

    21 引用 • 140 回帖 • 24 关注
  • Spring

    Spring 是一个开源框架,是于 2003 年兴起的一个轻量级的 Java 开发框架,由 Rod Johnson 在其著作《Expert One-On-One J2EE Development and Design》中阐述的部分理念和原型衍生而来。它是为了解决企业应用开发的复杂性而创建的。框架的主要优势之一就是其分层架构,分层架构允许使用者选择使用哪一个组件,同时为 JavaEE 应用程序开发提供集成的框架。

    940 引用 • 1458 回帖 • 159 关注
  • SpaceVim

    SpaceVim 是一个社区驱动的模块化 vim/neovim 配置集合,以模块的方式组织管理插件以
    及相关配置,为不同的语言开发量身定制了相关的开发模块,该模块提供代码自动补全,
    语法检查、格式化、调试、REPL 等特性。用户仅需载入相关语言的模块即可得到一个开箱
    即用的 Vim-IDE。

    3 引用 • 31 回帖 • 69 关注
  • Elasticsearch

    Elasticsearch 是一个基于 Lucene 的搜索服务器。它提供了一个分布式多用户能力的全文搜索引擎,基于 RESTful 接口。Elasticsearch 是用 Java 开发的,并作为 Apache 许可条款下的开放源码发布,是当前流行的企业级搜索引擎。设计用于云计算中,能够达到实时搜索,稳定,可靠,快速,安装使用方便。

    116 引用 • 99 回帖 • 266 关注
  • etcd

    etcd 是一个分布式、高可用的 key-value 数据存储,专门用于在分布式系统中保存关键数据。

    5 引用 • 26 回帖 • 493 关注
  • 以太坊

    以太坊(Ethereum)并不是一个机构,而是一款能够在区块链上实现智能合约、开源的底层系统。以太坊是一个平台和一种编程语言 Solidity,使开发人员能够建立和发布下一代去中心化应用。 以太坊可以用来编程、分散、担保和交易任何事物:投票、域名、金融交易所、众筹、公司管理、合同和知识产权等等。

    34 引用 • 367 回帖 • 2 关注
  • 书籍

    宋真宗赵恒曾经说过:“书中自有黄金屋,书中自有颜如玉。”

    76 引用 • 390 回帖 • 1 关注
  • 尊园地产

    昆明尊园房地产经纪有限公司,即:Kunming Zunyuan Property Agency Company Limited(简称“尊园地产”)于 2007 年 6 月开始筹备,2007 年 8 月 18 日正式成立,注册资本 200 万元,公司性质为股份经纪有限公司,主营业务为:代租、代售、代办产权过户、办理银行按揭、担保、抵押、评估等。

    1 引用 • 22 回帖 • 682 关注
  • 黑曜石

    黑曜石是一款强大的知识库工具,支持本地 Markdown 文件编辑,支持双向链接和关系图。

    A second brain, for you, forever.

    10 引用 • 85 回帖
  • 微信

    腾讯公司 2011 年 1 月 21 日推出的一款手机通讯软件。用户可以通过摇一摇、搜索号码、扫描二维码等添加好友和关注公众平台,同时可以将自己看到的精彩内容分享到微信朋友圈。

    129 引用 • 793 回帖
  • golang

    Go 语言是 Google 推出的一种全新的编程语言,可以在不损失应用程序性能的情况下降低代码的复杂性。谷歌首席软件工程师罗布派克(Rob Pike)说:我们之所以开发 Go,是因为过去 10 多年间软件开发的难度令人沮丧。Go 是谷歌 2009 发布的第二款编程语言。

    491 引用 • 1383 回帖 • 374 关注
  • Ant-Design

    Ant Design 是服务于企业级产品的设计体系,基于确定和自然的设计价值观上的模块化解决方案,让设计者和开发者专注于更好的用户体验。

    17 引用 • 23 回帖 • 1 关注
  • Facebook

    Facebook 是一个联系朋友的社交工具。大家可以通过它和朋友、同事、同学以及周围的人保持互动交流,分享无限上传的图片,发布链接和视频,更可以增进对朋友的了解。

    4 引用 • 15 回帖 • 454 关注
  • 自由行
    1 关注
  • SendCloud

    SendCloud 由搜狐武汉研发中心孵化的项目,是致力于为开发者提供高质量的触发邮件服务的云端邮件发送平台,为开发者提供便利的 API 接口来调用服务,让邮件准确迅速到达用户收件箱并获得强大的追踪数据。

    2 引用 • 8 回帖 • 437 关注
  • 开源中国

    开源中国是目前中国最大的开源技术社区。传播开源的理念,推广开源项目,为 IT 开发者提供了一个发现、使用、并交流开源技术的平台。目前开源中国社区已收录超过两万款开源软件。

    7 引用 • 86 回帖
  • JetBrains

    JetBrains 是一家捷克的软件开发公司,该公司位于捷克的布拉格,并在俄国的圣彼得堡及美国麻州波士顿都设有办公室,该公司最为人所熟知的产品是 Java 编程语言开发撰写时所用的集成开发环境:IntelliJ IDEA

    18 引用 • 54 回帖
  • 友情链接

    确认过眼神后的灵魂连接,站在链在!

    24 引用 • 373 回帖 • 4 关注
  • Ngui

    Ngui 是一个 GUI 的排版显示引擎和跨平台的 GUI 应用程序开发框架,基于
    Node.js / OpenGL。目标是在此基础上开发 GUI 应用程序可拥有开发 WEB 应用般简单与速度同时兼顾 Native 应用程序的性能与体验。

    7 引用 • 9 回帖 • 346 关注
  • SVN

    SVN 是 Subversion 的简称,是一个开放源代码的版本控制系统,相较于 RCS、CVS,它采用了分支管理系统,它的设计目标就是取代 CVS。

    29 引用 • 98 回帖 • 693 关注
  • PHP

    PHP(Hypertext Preprocessor)是一种开源脚本语言。语法吸收了 C 语言、 Java 和 Perl 的特点,主要适用于 Web 开发领域,据说是世界上最好的编程语言。

    164 引用 • 407 回帖 • 526 关注
  • 单点登录

    单点登录(Single Sign On)是目前比较流行的企业业务整合的解决方案之一。SSO 的定义是在多个应用系统中,用户只需要登录一次就可以访问所有相互信任的应用系统。

    9 引用 • 25 回帖 • 4 关注
  • WebClipper

    Web Clipper 是一款浏览器剪藏扩展,它可以帮助你把网页内容剪藏到本地。

    3 引用 • 9 回帖 • 4 关注
  • 资讯

    资讯是用户因为及时地获得它并利用它而能够在相对短的时间内给自己带来价值的信息,资讯有时效性和地域性。

    53 引用 • 85 回帖
  • V2Ray
    1 引用 • 15 回帖
  • MySQL

    MySQL 是一个关系型数据库管理系统,由瑞典 MySQL AB 公司开发,目前属于 Oracle 公司。MySQL 是最流行的关系型数据库管理系统之一。

    675 引用 • 535 回帖