看似简单的空指针 —— 包装类自动拆箱陷阱

一、Bug 场景

在一个 Java 应用程序中,涉及到一些基本数据类型与包装类的操作。开发人员在处理业务逻辑时,认为代码逻辑清晰简单,但在特定情况下,程序却抛出了空指针异常(NullPointerException),导致程序崩溃,影响了业务功能的正常运行。

二、代码示例

业务逻辑类(有缺陷)

java 复制代码
public class BusinessLogic {
    public static int calculateSum(Integer num1, Integer num2) {
        // 预期是对两个数求和,忽略了包装类可能为空的情况
        return num1 + num2; 
    }
}

测试代码

java 复制代码
public class NullPointerBugExample {
    public static void main(String[] args) {
        Integer a = null;
        Integer b = 5;
        try {
            int result = BusinessLogic.calculateSum(a, b);
            System.out.println("计算结果: " + result);
        } catch (NullPointerException e) {
            System.out.println("捕获到空指针异常: " + e.getMessage());
        }
    }
}

三、问题描述

  1. 预期行为calculateSum 方法应正确计算两个数的和并返回结果,即使其中一个参数为 null,也应按照业务逻辑处理,而不是抛出空指针异常。
  2. 实际行为 :当传入的 num1null 时,程序抛出 NullPointerException。这是因为在 Java 中,当对包装类(如 Integer)进行算术运算时,会发生自动拆箱,即将包装类转换为基本数据类型。在上述代码中,num1 + num2 时,num1 会自动拆箱为 int 类型。然而,null 无法转换为基本数据类型,因此抛出空指针异常。尽管代码表面上看起来只是简单的加法运算,但由于自动拆箱机制,隐藏了潜在的空指针风险。

四、解决方案

  1. 显式空值检查:在进行运算之前,对可能为空的包装类进行显式的空值检查。
java 复制代码
public class BusinessLogic {
    public static int calculateSum(Integer num1, Integer num2) {
        if (num1 == null) {
            num1 = 0;
        }
        if (num2 == null) {
            num2 = 0;
        }
        return num1 + num2; 
    }
}
  1. 使用 Optional 类(Java 8+)Optional 类提供了一种更优雅的方式来处理可能为空的值。
java 复制代码
import java.util.Optional;

public class BusinessLogic {
    public static int calculateSum(Optional<Integer> num1Opt, Optional<Integer> num2Opt) {
        int num1 = num1Opt.orElse(0);
        int num2 = num2Opt.orElse(0);
        return num1 + num2; 
    }
}

测试代码调整为:

java 复制代码
import java.util.Optional;

public class NullPointerBugExample {
    public static void main(String[] args) {
        Optional<Integer> a = Optional.ofNullable(null);
        Optional<Integer> b = Optional.of(5);
        int result = BusinessLogic.calculateSum(a, b);
        System.out.println("计算结果: " + result);
    }
}
相关推荐
qq_547026179几秒前
Maven 使用指南
java·maven
xiaolyuh1239 分钟前
Arthas修改类(如加日志)的实现原理
java
栗子叶13 分钟前
Java对象创建的过程
java·开发语言·jvm
有一个好名字23 分钟前
力扣-从字符串中移除星号
java·算法·leetcode
zfj32132 分钟前
CyclicBarrier、CountDownLatch、Semaphore 各自的作用和用法区别
java·开发语言·countdownlatch·semaphore·cyclicbarrier
2501_9167665438 分钟前
【JVM】类的加载机制
java·jvm
Sag_ever39 分钟前
Java数组详解
java
张np39 分钟前
java基础-ConcurrentHashMap
java·开发语言
一嘴一个橘子1 小时前
spring-aop 的 基础使用 - 4 - 环绕通知 @Around
java
小毅&Nora1 小时前
【Java线程安全实战】⑨ CompletableFuture的高级用法:从基础到高阶,结合虚拟线程
java·线程安全·虚拟线程