0

0

Java JUnit assertThrows 与异常消息比对:避免常见陷阱

霞舞

霞舞

发布时间:2025-10-09 12:06:46

|

764人浏览过

|

来源于php中文网

原创

Java JUnit assertThrows 与异常消息比对:避免常见陷阱

本教程深入探讨在 Java JUnit 中使用 assertThrows 进行异常测试时常见的 assertEquals 误用问题。它详细解释了为何不能直接将期望的字符串消息与捕获到的异常对象进行比对,并提供了正确的解决方案:通过 e.getMessage() 获取异常消息进行精确断言,确保测试的准确性和健壮性。

软件开发中,正确地处理异常是构建健壮应用的关键一环。单元测试,特别是使用 junit 框架,为我们验证异常行为提供了强大的工具。其中,assertthrows 方法是专门用于断言某个操作会抛出特定类型异常的。然而,在使用 assertthrows 捕获到异常后,如何正确地比对异常消息,常常会成为初学者遇到的一个困惑点。

理解 assertThrows

JUnit 5 提供的 assertThrows 方法是一个非常有用的工具,它允许我们验证代码是否按预期抛出异常。它的基本用法如下:

ExceptionType actualException = assertThrows(
    ExceptionType.class, // 期望抛出的异常类型
    () -> { /* 会抛出异常的代码块 */ }, // 可执行的 Lambda 表达式
    "断言失败时的消息(可选)" // 当未抛出异常或抛出类型不匹配时显示的消息
);

assertThrows 方法会执行提供的 Lambda 表达式。如果表达式抛出了指定类型的异常,该方法就会返回这个异常实例;否则,测试将失败。

常见的错误:直接比对字符串与异常对象

一个常见的错误是在捕获到异常后,尝试直接将期望的异常消息字符串与 assertThrows 返回的异常对象进行 assertEquals 比对。

考虑以下一个简单的 Calculator 类,其中 multiply 方法在乘数为零时抛出 IllegalArgumentException:

立即学习Java免费学习笔记(深入)”;

// Calculator.java
public class Calculator {
    public int multiply(int i, int j) {
        if (i == 0 || j == 0) {
            throw new IllegalArgumentException("* by zero");
        }
        return i * j;
    }
}

现在,假设我们编写一个 JUnit 测试来验证当其中一个乘数为零时是否抛出正确的异常,并期望其消息为 "* by zero"。一个错误的测试尝试可能如下:

import org.junit.jupiter.api.Test;
import static org.junit.jupiter.api.Assertions.*;

public class CalculatorTest {

    private Calculator tm = new Calculator(); // 假设 tm 是 Calculator 的实例

    @Test
    void testMultiply_WhenZeroIsMultiplied_ShouldThrowException() {
        int i = 0;
        int j = 4;
        String expectedMsg = "* by zero";

        // 使用 assertThrows 捕获异常
        IllegalArgumentException e = assertThrows(
                IllegalArgumentException.class,
                () -> tm.multiply(i, j),
                "期望在乘数为零时抛出 IllegalArgumentException"
        );

        // 错误的断言方式:直接比较 String 和 Exception 对象
        // assertEquals("断言错误", expectedMsg, e); // 编译错误或运行时错误
    }
}

如果直接使用 assertEquals("断言错误", expectedMsg, e);,会遇到类型不匹配的问题。expectedMsg 是一个 String 类型,而 e 是一个 IllegalArgumentException 类型的对象。assertEquals 方法没有直接用于比较 String 和 Exception 对象的重载。即使尝试将异常对象作为期望值(如 assertEquals("Error", new IllegalArgumentException("* by zero"), e);),assertEquals 默认也是通过对象的 equals() 方法进行比较。对于大多数异常类,其 equals() 方法继承自 Object,默认比较的是对象的引用地址,而不是其内容(如消息)。因此,即使消息相同,两个不同的异常对象实例也会被认为是不同的。

CA.LA
CA.LA

第一款时尚产品在线设计平台,服装设计系统

下载

正确的解决方案:比对异常消息

要正确地比对异常消息,我们需要从捕获到的异常对象中提取出其消息字符串,然后与我们期望的字符串进行比较。所有 Throwable 类都提供了 getMessage() 方法来获取异常的详细消息。

修正后的测试代码应如下所示:

import org.junit.jupiter.api.Test;
import static org.junit.jupiter.api.Assertions.*;

public class CalculatorTest {

    private Calculator tm = new Calculator(); // 假设 tm 是 Calculator 的实例

    @Test
    void testMultiply_WhenZeroIsMultiplied_ShouldThrowException() {
        int i = 0;
        int j = 4;
        String expectedMsg = "* by zero";

        // 使用 assertThrows 捕获异常
        IllegalArgumentException e = assertThrows(
                IllegalArgumentException.class,
                () -> tm.multiply(i, j),
                "期望在乘数为零时抛出 IllegalArgumentException"
        );

        // 正确的断言方式:比较期望消息字符串与异常对象的getMessage()结果
        assertEquals(expectedMsg, e.getMessage(), "异常消息不匹配");
        // 或者,如果你更喜欢使用 assertTrue 结合 equals 方法:
        // assertTrue(e.getMessage().equals(expectedMsg), "异常消息不匹配");
    }
}

在这个修正后的代码中,assertEquals(expectedMsg, e.getMessage(), "异常消息不匹配"); 能够正确工作,因为它现在比较的是两个 String 对象:expectedMsg 和 e.getMessage() 的返回值。

示例代码

为了提供一个完整的示例,我们结合 Calculator 类和修正后的测试类:

// src/main/java/com/example/Calculator.java
package com.example;

public class Calculator {
    public int multiply(int i, int j) {
        if (i == 0 || j == 0) {
            throw new IllegalArgumentException("* by zero");
        }
        return i * j;
    }
}
// src/test/java/com/example/CalculatorTest.java
package com.example;

import org.junit.jupiter.api.Test;
import static org.junit.jupiter.api.Assertions.*;

public class CalculatorTest {

    private Calculator calculator = new Calculator();

    @Test
    void testMultiply_WhenFirstFactorIsZero_ShouldThrowExceptionWithMessage() {
        int i = 0;
        int j = 4;
        String expectedMsg = "* by zero";

        // 1. 使用 assertThrows 验证是否抛出指定类型的异常
        IllegalArgumentException actualException = assertThrows(
                IllegalArgumentException.class,
                () -> calculator.multiply(i, j),
                "期望在第一个乘数为零时抛出 IllegalArgumentException"
        );

        // 2. 验证异常消息是否与期望值匹配
        assertEquals(expectedMsg, actualException.getMessage(), "异常消息应为 '* by zero'");
    }

    @Test
    void testMultiply_WhenSecondFactorIsZero_ShouldThrowExceptionWithMessage() {
        int i = 5;
        int j = 0;
        String expectedMsg = "* by zero";

        IllegalArgumentException actualException = assertThrows(
                IllegalArgumentException.class,
                () -> calculator.multiply(i, j),
                "期望在第二个乘数为零时抛出 IllegalArgumentException"
        );

        assertEquals(expectedMsg, actualException.getMessage(), "异常消息应为 '* by zero'");
    }

    @Test
    void testMultiply_WhenNoZeroFactor_ShouldReturnCorrectResult() {
        int i = 5;
        int j = 4;
        int expectedResult = 20;

        int actualResult = calculator.multiply(i, j);

        assertEquals(expectedResult, actualResult, "非零乘数相乘应返回正确结果");
    }
}

注意事项与最佳实践

  1. 区分对象与消息: 始终记住 assertThrows 返回的是异常 对象 本身,而不是它的消息字符串。要获取消息,必须调用 getMessage() 方法。
  2. assertEquals 的类型匹配: assertEquals 方法有多个重载,它会根据你传入的参数类型来选择最合适的版本。确保你比较的两个参数是类型兼容的(例如,都是 String)。
  3. 断言失败消息: 在 assertEquals 和 assertThrows 中提供清晰的断言失败消息(作为最后一个参数)是一个好习惯。这在测试失败时能帮助你更快地定位问题。
  4. 精确性与健壮性: 除了检查异常类型和消息外,有时你可能还需要检查异常的其他属性,例如 getCause() 或自定义异常中的特定字段。确保你的测试覆盖了所有重要的异常行为。
  5. 避免过度测试: 虽然测试异常行为很重要,但也要避免过度测试。只测试那些关键的、容易出错的异常路径。

总结

在 JUnit 中进行异常测试时,assertThrows 是一个强大的工具,但正确比对异常消息是确保测试有效性的关键。核心原则是:assertThrows 返回的是异常对象,要比对其消息,必须通过 getMessage() 方法获取字符串,再与期望的字符串进行比较。遵循这些指导原则,可以帮助你编写出更健壮、更清晰的异常处理单元测试。

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

845

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

745

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

740

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

397

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

420

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

447

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

431

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16947

2023.08.03

c++ 根号
c++ 根号

本专题整合了c++根号相关教程,阅读专题下面的文章了解更多详细内容。

45

2026.01.23

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 2.8万人学习

C# 教程
C# 教程

共94课时 | 7.5万人学习

Java 教程
Java 教程

共578课时 | 50.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号