如何在Java中判断是否为空

更多内容关注公众号:SAP Technical 

NullPointerException

作为搬砖党的一族们,我们对判空一定再熟悉不过了,不要跟我说你很少进行判空,除非你喜欢NullPointerException。

不过NullPointerException对于很多猿们来说,也是Exception家族中最亲近的一员了。

为了避免NullPointerException来找我们,我们经常会进行如下操作。

if (data != null) {
    do sth.
}

如果一个类中多次使用某个对象,那你可能要一顿操作,so:

1

“世界第九大奇迹”就这样诞生了。Maybe你会想,项目中肯定不止你一个人会这样一顿操作,然后按下Command+Shift+F,真相就在眼前:

2

What,我们有接近一万行的代码都是在判空?

好了,接下来,要进入正题了。

NullObject模式

对于项目中无数次的判空,对代码质量整洁度产生了十分之恶劣的影响,对于这种现象,我们称之为“判空灾难”。

那么,这种现象如何治理呢,你可能听说过NullObject模式,不过这不是我们今天的武器,但是还是需要介绍一下NullObject模式。

什么是NullObject模式呢?

In object-oriented computer programming, a null object is an object with no referenced value or with defined neutral ("null") behavior. The null object design pattern describes the uses of such objects and their behavior (or lack thereof).

以上解析来自Wikipedia。

NullObject模式首次发表在“ 程序设计模式语言 ”系列丛书中。一般的,在面向对象语言中,对对象的调用前需要使用判空检查,来判断这些对象是否为空,因为在空引用上无法调用所需方法。

空对象模式的一种典型实现方式如下图所示(图片来自网络):

4

示例代码如下(命名来自网络,哈哈到底是有多懒):

Nullable是空对象的相关操作接口,用于确定对象是否为空,因为在空对象模式中,对象为空会被包装成一个Object,成为Null Object,该对象会对原有对象的所有方法进行空实现。。

public interface Nullable {
    
    boolean isNull();
    
}

这个接口定义了业务对象的行为。

public interface DependencyBase extends Nullable {

    void Operation();

}

这是该对象的真实类,实现了业务行为接口DependencyBase与空对象操作接口Nullable。

public class Dependency implements DependencyBase, Nullable {

    @Override
    public void Operation() {
        System.out.print("Test!");
    }

    @Override
    public boolean isNull() {
        return false;
    }
    
}

这是空对象,对原有对象的行为进行了空实现。

public class NullObject implements DependencyBase{

    @Override
    public void Operation() {
        // do nothing
    }

    @Override
    public boolean isNull() {
        return true;
    }
    
}

在使用时,可以通过工厂调用方式来进行空对象的调用,也可以通过其他如反射的方式对对象进行调用(一般多耗时几毫秒)在此不进行详细叙述。

public class Factory {
    
    public static DependencyBase get(Nullable dependencyBase){
        if (dependencyBase == null){
            return new NullObject();
        }
        return new Dependency();
    }
    
}

这是一个使用范例,通过这种模式,我们不再需要进行对象的判空操作,而是可以直接使用对象,也不必担心NPE(NullPointerException)的问题。

public class Client {

    public void test(DependencyBase dependencyBase){
        Factory.get(dependencyBase).Operation();
    }

}

关于空对象模式,更具体的内容大家也可以多找一找资料,上述只是对NullObject的简单介绍,但是,今天我要推荐的是一款协助判空的插件NR Null Object,让我们来优雅地进行判空,不再进行一顿操作来定义繁琐的空对象接口与空独享实现类。

.NR Null Object

NR Null Object是一款适用于Android Studio、IntelliJ IDEA、PhpStorm、WebStorm、PyCharm、RubyMine、AppCode、CLion、GoLand、DataGrip等IDEA的Intellij插件。其可以根据现有对象,便捷快速生成其空对象模式需要的组成成分,其包含功能如下:

  • 分析所选类可声明为接口的方法;
  • 抽象出公有接口;
  • 创建空对象,自动实现公有接口;
  • 对部分函数进行可为空声明;
  • 可追加函数进行再次生成;
  • 自动的函数命名规范

让我们来看一个使用范例:

5

怎么样,看起来是不是非常快速便捷,只需要在原有需要进行多次判空的对象中,邮件弹出菜单,选择Generate,并选择NR Null Object即可自动生成相应的空对象组件。

那么如何来获得这款插件呢?

安装方式

可以直接通过IDEA的Preferences中的Plugins仓库进行安装。

选择 Preferences → Plugins → Browse repositories

6

搜索“NR Null Oject”或者“Null Oject”进行模糊查询,点击右侧的Install,restart IDEA即可。

7

Optional

还有一种方式是使用Java8特性中的Optional来进行优雅地判空,Optional来自官方的介绍如下:

A container object which may or may not contain a non-null value. If a value is present, isPresent() will return true and get() will return the value.

一个可能包含也可能不包含非null值的容器对象。 如果存在值,isPresent()将返回true,get()将返回该值。

话不多说,举个例子。

有如下代码,需要获得Test2中的Info信息,但是参数为Test4,我们要一层层的申请,每一层都获得的对象都可能是空,最后的代码看起来就像这样。

    public String testSimple(Test4 test) {
        if (test == null) {
            return "";
        }
        if (test.getTest3() == null) {
            return "";
        }
        if (test.getTest3().getTest2() == null) {
            return "";
        }
        if (test.getTest3().getTest2().getInfo() == null) {
            return "";
        }
        return test.getTest3().getTest2().getInfo();
    }

但是使用Optional后,整个就都不一样了。

    public String testOptional(Test test) {
        return Optional.ofNullable(test).flatMap(Test::getTest3)
                .flatMap(Test3::getTest2)
                .map(Test2::getInfo)
                .orElse("");
    }

1.Optional.ofNullable(test),如果test为空,则返回一个单例空Optional对象,如果非空则返回一个Optional包装对象,Optional将test包装;

    public static <T> Optional<T> ofNullable(T value) {
        return value == null ? empty() : of(value);
    }

2.flatMap(Test::getTest3)判断test是否为空,如果为空,继续返回第一步中的单例Optional对象,否则调用Test的getTest3方法;

    public<U> Optional<U> flatMap(Function<? super T, Optional<U>> mapper) {
        Objects.requireNonNull(mapper);
        if (!isPresent())
            return empty();
        else {
            return Objects.requireNonNull(mapper.apply(value));
        }
    }

3.flatMap(Test3::getTest2)同上调用Test3的getTest2方法;

4.map(Test2::getInfo)同flatMap类似,但是flatMap要求Test3::getTest2返回值为Optional类型,而map不需要,flatMap不会多层包装,map返回会再次包装Optional;

    public<U> Optional<U> map(Function<? super T, ? extends U> mapper) {
        Objects.requireNonNull(mapper);
        if (!isPresent())
            return empty();
        else {
            return Optional.ofNullable(mapper.apply(value));
        }
    }

5.orElse("");获得map中的value,不为空则直接返回value,为空则返回传入的参数作为默认值。

public T orElse(T other) {
    return value != null ? value : other;
}

怎么样,使用Optional后我们的代码是不是瞬间变得非常整洁,或许看到这段代码你会有很多疑问,针对复杂的一长串判空,Optional有它的优势,但是对于简单的判空使用Optional也会增加代码的阅读成本、编码量以及团队新成员的学习成本。毕竟Optional在现在还并没有像RxJava那样流行,它还拥有一定的局限性。

如果直接使用Java8中的Optional,需要保证安卓API级别在24及以上。

image-20181124085913887.png

你也可以直接引入Google的Guava。(啥是Guava?来自官方的提示)

Guava is a set of core libraries that includes new collection types (such as multimap and multiset), immutable collections, a graph library, functional types, an in-memory cache, and APIs/utilities for concurrency, I/O, hashing, primitives, reflection, string processing, and much more!

引用方式,就像这样:

    dependencies {
      compile 'com.google.guava:guava:27.0-jre'
      // or, for Android:
      api 'com.google.guava:guava:27.0-android'
    }

不过IDEA默认会显示黄色,提示让你将Guava表达式迁移到Java Api上。

image-3.png

当然,你也可以通过在Preferences搜索"Guava"来Kill掉这个Yellow的提示。

image-4.png

关于Optional使用还有很多技巧,感兴趣可以查阅Guava和Java8相关书籍和文档。

使用Optional具有如下优点:

  1. 将防御式编程代码完美包装
  2. 链式调用
  3. 有效避免程序代码中的空指针

但是也同样具有一些缺点:

  1. 流行性不是非常理想,团队新成员需要学习成本
  2. 安卓中需要引入Guava,需要团队每个人处理IDEA默认提示,或者忍受黄色提示

Kotlin

当然,Kotlin以具有优秀的空安全性为一大特色,并可以与Java很好的混合使用,like this:

    test1?.test2?.test3?.test4

如果你已经开始使用了Kotlin,可以不用再写缭乱的防御判空语句。如果你还没有使用Kotlin,并不推荐为了判空优雅而直接转向Kotlin。

<think> 首先,用户的问题是:“我想查找在Java判断对象是否的方法”,意思是“I want to find methods to determine if an object is null in Java”。 我有几个引用: - 引用[1]:描述了Java判断对象是否的不同方法,包括使用条件判断、Optional类、Apache Commons Lang库等。 - 引用[2]:类似,讨论了使用==、Objects.isNull()、instanceof、Optional类。 - 引用[3]:提到了StringUtils.isEmpty用于字符串。 - 引用[4]:针对List判断的方法,推荐使用isEmpty()。 - 引用[5]:涉及StringUtils.isEmpty。 用户的问题是关于判断对象是否,所以我需要覆盖Java中处理null对象的一般方法。 系统指令: - 所有行内数学表达式必须用$...$,但这里不涉及数学,所以可能不需要。 - 独立公式用$$...$$,但同样不相关。 - LaTeX语法正确:确保任何数学部分正确,但这里没有数学。 - 使用中文回答:所有响应必须用中文。 - 生成相关问题:在回答末尾,添加相关问题部分。 - 引用中引用的段落末尾添加引用标识,例如[^1]。 - 用户可见层:回答结构清晰,逐步解决问题,确保真实可靠。 回答结构: - 以清晰的方式解释方法。 - 因为这是技术问题,提供代码示例会更好。 - 参考引用来支持信息。 - 在末尾添加相关问题。 关键方法从引用中提取: 1. 使用==或!=:直接比较对象和null。 2. 使用Objects.isNull()Java 7引入,安全处理null。 3. 使用Optional类:Java 8引入,优雅处理。 4. 使用Apache Commons Lang库的StringUtils或类似工具。 5. 对于集合如List,使用isEmpty()或size()==0。 确保覆盖一般对象和特定类型如字符串或集合。 步骤: 1. 介绍问题:指针异常的常见错误,判断对象的重要性。 2. 逐步列出方法: - 基本方法:使用== null - 使用Objects.isNull() - 使用Optional - 使用库如Apache Commons - 针对字符串和集合的特定方法 3. 提供代码示例。 4. 添加引用标识。 5. 结束部分添加相关问题。 相关问题生成: - 基于主题,生成3-4个相关问题。 - 例如:如何避免指针异常?Java 8 Optional的更多用法?Apache Commons库的使用?等。 最终输出: - 用中文。 - 结构:解释、方法列表、代码示例、相关问题。 </think> ### Java判断对象是否的方法详解 在Java编程中,判断对象是否是防止指针异常(NullPointerException)的关键步骤。指针异常是常见的程序错误,会导致应用崩溃或不可预测行为。正确判断对象值能提高代码健壮性和可维护性。以下是Java判断对象是否的主要方法,我将逐步解释每种方法的原理、优缺点和代码示例。 #### 1. **基本方法:使用`== null`或`!= null`** - **原理**:直接比较对象引用与`null`值。这是最基础且高效的方式,适用于所有对象类型。 - **优点**:简单直接,无需额外依赖。 - **缺点**:如果对象为`null`时调用其方法,仍可能引发指针异常。 - **代码示例**: ```java Object obj = null; if (obj == null) { System.out.println("对象为"); // 输出:对象为 } else { System.out.println("对象不为"); } ``` 这种方法在判断字符串或自定义对象时通用[^2]。 #### 2. **使用`Objects.isNull()`方法(Java 7及以上)** - **原理**:`java.util.Objects`类提供的静态方法,安全地检查对象是否为`null`。它内部使用`obj == null`实现,但能避免某些边界问题。 - **优点**:代码更易读,且能处理`null`参数而不抛异常。 - **缺点**:需要Java 7或更高版本。 - **代码示例**: ```java import java.util.Objects; Object obj = null; if (Objects.isNull(obj)) { System.out.println("对象为"); // 输出:对象为 } else { System.out.println("对象不为"); } ``` 这个方法特别适合在lambda表达式或流处理中使用[^2]。 #### 3. **使用`Optional`类(Java 8及以上)** - **原理**:`java.util.Optional`类封装对象,提供优雅的值处理。`Optional.ofNullable()`创建Optional实例,`isPresent()`检查对象是否存在。 - **优点**:支持链式调用,减少嵌套if语句,提升代码可读性。 - **缺点**:引入额外对象开销,适用于复杂场景。 - **代码示例**: ```java import java.util.Optional; Object obj = null; Optional<Object> optional = Optional.ofNullable(obj); if (optional.isPresent()) { System.out.println("对象不为"); } else { System.out.println("对象为"); // 输出:对象为 } ``` 结合`orElse()`或`orElseGet()`方法,可以设置默认值[^2]。 #### 4. **使用第三方库(如Apache Commons Lang)** - **原理**:库如Apache Commons Lang提供工具类(如`StringUtils.isEmpty()`)简化值检查。 - **针对字符串**:`StringUtils.isEmpty(CharSequence cs)`检查字符串为`null`或串。 - **针对其他对象**:自定义工具方法扩展。 - **优点**:减少样板代码,支持多类型检查。 - **缺点**:需引入外部依赖。 - **代码示例**: ```java import org.apache.commons.lang3.StringUtils; String str = null; if (StringUtils.isEmpty(str)) { System.out.println("字符串为"); // 输出:字符串为 } else { System.out.println("字符串不为"); } ``` 对于非字符串对象,可扩展类似工具类[^3][^5]。 #### 5. **针对特定类型的方法** - **集合类型(如List)**:优先使用`isEmpty()`而非`size() == 0`,因为`isEmpty()`语义更清晰。 ```java import java.util.List; import java.util.ArrayList; List<String> list = new ArrayList<>(); if (list.isEmpty()) { // 推荐使用 System.out.println("List为"); } ``` `size() == 0`在集合为时也有效,但`isEmpty()`更易读[^4]。 - **数组**:需单独检查长度,例如`if (array == null || array.length == 0)`。 #### 最佳实践建议 - **优先选择**:对于简单场景,使用`Objects.isNull()`或`== null`;对于复杂逻辑,使用`Optional`。 - **避免错误**:不要依赖`instanceof`作为值检查工具,因为它只在对象非`null`时有效。 - **性能注意**:基本方法`== null`性能最优,`Optional`或库方法稍慢但可读性高。 - **综合示例**: ```java public class NullCheckDemo { public static void main(String[] args) { Object obj = null; // 使用Objects.isNull() if (Objects.isNull(obj)) { System.out.println("Objects.isNull()检测为"); } // 使用Optional Optional.ofNullable(obj).ifPresentOrElse( o -> System.out.println("Optional检测不为"), () -> System.out.println("Optional检测为") ); } } ``` 通过以上方法,您可以有效避免指针异常。记住,选择方法时应考虑代码上下文和Java版本兼容性[^1][^2]。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值