0

0

JUnit中如何测试方法抛出特定异常及自定义消息

碧海醫心

碧海醫心

发布时间:2025-10-04 11:03:01

|

538人浏览过

|

来源于php中文网

原创

JUnit中如何测试方法抛出特定异常及自定义消息

本文旨在指导开发者如何在JUnit测试中验证方法是否抛出特定类型的异常,并检查异常携带的自定义消息。我们将探讨JUnit 4中@Test(expected=...)注解的使用方法及其局限性,并详细介绍通过try-catch块实现更灵活、可检查异常内容的测试策略,同时简要提及JUnit 5的现代做法。

软件开发中,确保代码在异常情况下能够正确抛出预期的异常是健壮性测试的重要组成部分。特别是在处理用户输入、资源访问或业务逻辑校验时,我们经常需要验证方法在遇到无效状态时是否抛出特定类型且带有明确信息的异常。

1. 待测试方法示例

为了更好地说明异常测试,我们首先定义一个简单的Java方法,该方法在特定条件下会抛出ArithmeticException:

public class CalculatorUtils {

    /**
     * 计算数组中给定数字的倍数数量。
     * 如果给定数字为零,则抛出ArithmeticException。
     *
     * @param number 用于检查倍数的数字。
     * @param array  待检查的整数数组。
     * @return 数组中给定数字的倍数数量。
     */
    public static int getMultiplesOfGivenNumber(int number, int[] array) {
        if (number == 0) {
            throw new ArithmeticException("Number cannot be zero");
        } else {
            int multiples = 0;
            for (int i = 0; i < array.length; i++) {
                if (array[i] % number == 0) {
                    multiples += 1;
                }
            }
            return multiples;
        }
    }

    // 辅助方法,用于生成测试数据
    public static int[] intervalFromOneToTen() {
        return new int[]{1, 2, 3, 4, 5, 6, 7, 8, 9, 10};
    }
}

我们的目标是测试当number参数为0时,getMultiplesOfGivenNumber方法是否正确抛出ArithmeticException,并验证其错误消息是否为"Number cannot be zero"。

2. 使用 @Test(expected=...) 注解进行异常测试 (JUnit 4)

JUnit 4 提供了一个方便的注解参数expected,可以直接在@Test注解中使用,用于声明一个测试方法预期会抛出的异常类型。

2.1 使用方法

import org.junit.Test;
import static org.junit.Assert.fail; // 如果异常未抛出,可使用fail()

public class CalculatorUtilsTest {

    // 辅助方法,用于生成测试数据
    private int[] intervalFromOneToTen() {
        return new int[]{1, 2, 3, 4, 5, 6, 7, 8, 9, 10};
    }

    @Test(expected = ArithmeticException.class)
    public void testDivideByZeroWithExpectedAnnotation() {
        // Arrange
        int number = 0;
        // Act
        // 调用可能抛出异常的方法。如果ArithmeticException被抛出,测试通过。
        // 如果未抛出异常,或者抛出了不同类型的异常,测试将失败。
        CalculatorUtils.getMultiplesOfGivenNumber(number, intervalFromOneToTen());
        // 注意:在此处不需要额外的断言,因为@Test(expected)本身就是一种断言。
        // 如果代码执行到这里而没有抛出异常,JUnit会自动使测试失败。
    }
}

2.2 注意事项与局限性

  • JUnit 版本要求:@Test(expected=...)特性仅在 JUnit 4 及更高版本中可用。如果你在使用早期版本的JUnit,你的IDE将无法识别expected参数。
  • 无法检查异常消息:这是@Test(expected=...)的主要局限性。它只能验证异常的类型,而无法获取异常对象本身来检查其详细信息,例如异常消息、原因(cause)或任何自定义字段。
  • 测试通过条件:只要指定类型的异常在方法执行过程中被抛出,测试就会通过。如果方法执行完毕而未抛出异常,或者抛出了不同类型的异常,测试将失败。

3. 使用 try-catch 块进行异常测试 (更灵活,兼容性好)

try-catch块是一种更通用、更灵活的异常测试方法,它不仅适用于所有JUnit版本,而且允许你完全控制异常对象,从而可以验证其消息、原因及其他属性。

3.1 使用方法

import org.junit.Test;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.fail; // 用于断言异常未抛出时测试失败

public class CalculatorUtilsTest {

    // 辅助方法,用于生成测试数据
    private int[] intervalFromOneToTen() {
        return new int[]{1, 2, 3, 4, 5, 6, 7, 8, 9, 10};
    }

    @Test
    public void testDivideByZeroWithTryCatch() {
        // Arrange
        int number = 0;
        try {
            // Act
            CalculatorUtils.getMultiplesOfGivenNumber(number, intervalFromOneToTen());
            // 如果代码执行到这里,说明预期异常没有被抛出,测试应该失败。
            fail("Expected ArithmeticException was not thrown.");
        } catch (ArithmeticException e) {
            // Assert
            // 捕获到预期的ArithmeticException,现在可以检查其详细信息。
            assertEquals("Number cannot be zero", e.getMessage());
            // 如果有其他需要检查的异常属性,也可以在此处进行断言。
        } catch (Exception e) {
            // 如果抛出了不同类型的异常,也应该失败。
            fail("Expected ArithmeticException, but caught a different exception: " + e.getClass().getSimpleName());
        }
    }
}

3.2 优势

  • 全面检查异常内容:你可以获取到异常对象e,并对其所有可访问的属性(如getMessage()、getCause()、自定义字段等)进行断言。这对于验证异常消息是否符合预期至关重要。
  • 兼容性强:这种方法不依赖于特定的JUnit版本,在所有JUnit版本中均可使用。
  • 更精确的控制:你可以区分未抛出异常、抛出正确类型但错误消息的异常,以及抛出完全不同类型异常的情况。
  • 处理多种异常:如果一个方法可能抛出多种不同类型的异常,你可以使用多个catch块来分别处理和验证。

3.3 fail() 方法的使用

在try块的末尾,如果代码执行到该位置而没有抛出预期的异常,我们使用fail()方法来明确地使测试失败。fail()方法会抛出一个AssertionError,其中包含你提供的消息,清晰地表明测试未达到预期。

文心快码
文心快码

文心快码(Comate)是百度推出的一款AI辅助编程工具

下载

4. JUnit 5 的现代异常测试方法

对于使用 JUnit 5 的项目,推荐使用Assertions.assertThrows()方法进行异常测试。它提供了一种更简洁、更功能性的方式来验证异常。

import org.junit.jupiter.api.Test;
import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.assertThrows;

public class CalculatorUtilsTest {

    // 辅助方法,用于生成测试数据
    private int[] intervalFromOneToTen() {
        return new int[]{1, 2, 3, 4, 5, 6, 7, 8, 9, 10};
    }

    @Test
    void testDivideByZeroWithAssertThrows() {
        // 使用assertThrows捕获异常,并返回异常对象
        ArithmeticException exception = assertThrows(ArithmeticException.class, () -> {
            CalculatorUtils.getMultiplesOfGivenNumber(0, intervalFromOneToTen());
        });

        // 现在可以对捕获到的异常对象进行断言
        assertEquals("Number cannot be zero", exception.getMessage());
    }
}

assertThrows()方法接收两个参数:预期的异常类型和执行可能抛出异常的代码的Lambda表达式。它会返回捕获到的异常实例,使得后续对异常属性的断言变得非常方便和直观。

总结

在JUnit中测试方法抛出异常是确保代码健壮性的关键步骤。

  • 对于 JUnit 4,你可以选择使用简洁但功能有限的@Test(expected=...)注解来验证异常类型。
  • 如果需要检查异常的详细信息(如消息),或者你的JUnit版本不支持expected参数,那么 try-catch块是更强大、更灵活的选择。
  • 对于 JUnit 5,推荐使用Assertions.assertThrows()方法,它结合了简洁性与功能性,是现代Java测试的首选。

无论选择哪种方法,核心目标都是确保在特定条件下,你的代码能够按照预期抛出正确类型的异常,并携带准确的错误信息。

相关专题

更多
java
java

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

825

2023.06.15

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

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

724

2023.07.05

java自学难吗
java自学难吗

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

728

2023.07.31

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

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

395

2023.08.01

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

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

398

2023.08.02

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

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

445

2023.08.02

java有什么用
java有什么用

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

428

2023.08.02

java在线网站
java在线网站

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

16861

2023.08.03

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

7

2025.12.31

热门下载

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

精品课程

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

共23课时 | 2.1万人学习

C# 教程
C# 教程

共94课时 | 5.7万人学习

Java 教程
Java 教程

共578课时 | 39.9万人学习

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

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