《Java单元测试实战》——无效单测:那些年,我们写过的无效单元测试(10)

简介: 《Java单元测试实战》——无效单测:那些年,我们写过的无效单元测试(10)

《Java单元测试实战》——无效单测:那些年,我们写过的无效单元测试(9) https://developer.aliyun.com/article/1232106?groupCode=java



3. 验证抛出异常问题

 

这里,以创建用户时抛出异常为例,来说明验证抛出异常时所存在的问题。

 

代码案例:


image.png

1) 不验证抛出异常类型

 

反面案例:

 

在验证抛出异常时,很多人使用@Test注解的expected属性,并且指定取值为Exception.class,主要原因是:

 

单元测试用例的代码简洁,只有一行@Test注解;

不管抛出什么异常,都能保证单元测试用例通过。

image.png

 

 存在问题:

 

上面用例指定了通用异常类型,没有对抛出异常类型进行验证。所以,如果把ExampleException异常改为RuntimeException异常,该单元测试用例是无法验证出来的。

image.png

2) 不验证抛出异常属性

 

反面案例:

 

既然需要验证异常类型,简单地指定@Test注解的expected属性为ExampleException.class即可。

image.png

存在问题:

 

上面用例只验证了异常类型,没有对抛出异常属性字段(异常消息、异常原因、错误编码等)进行验证。所以,如果把错误编码DATABASE_ERROR(数据库错误)改为PARAMETER_ERROR(参数错误),该单元测试用例是无法验证出来的。

image.png


3) 只验证抛出异常部分属性

 

反面案例:

 

如果要验证异常属性,就必须用Assert.assertThrows方法捕获异常,并对异常的常用属性进行验证。但是,有些人为了偷懒,只验证抛出异常部分属性。

 

image.png

存在问题:

 

上面用例只验证了异常类型和错误编码,如果把错误消息创建用户异常改为“创建用户错误”,该单元测试用例是无法验证出来的。


image.png

4) 不验证抛出异常原因

 

反面案例:

 

先捕获抛出异常,再验证异常编码和异常消息,看起来很完美了。

 

image.png

存在问题:

 

通过代码可以看出,在抛出ExampleException异常时,最后一个参数e是我们模拟的userService.createUser方法抛出的RuntimeException异常。但是,我们没有对抛出异常原因进行验证。如果修改代码,把最后一个参数e去掉,上面的单元测试用例是无法验证出来的。

image.png

5) 不验证相关方法调用

 

反面案例:

 

很多人认为,验证抛出异常就只验证抛出异常,验证依赖方法调用不是必须的。


image.png

存在问题:

 

如果不验证相关方法调用,如何能证明代码走过这个分支?比如:我们在创建用户之前,检查用户名称无效并抛出异常。

 

image.png


6) 完美地验证抛出异常

 

一个完美的异常验证,除对异常类型、异常属性、异常原因等进行验证外,还需对抛出异常前的依赖方法调用进行验证。

 

 完美案例:


image.png


4. 验证抛出异常准则

 

1) 必须验证所有抛出异常

 

在单元测试中,必须验证所有抛出异常:

 

来源于属性字段的判断

来源于输入参数的判断

来源于返回值的判断

来源于模拟方法的调用

来源于静态方法的调用

 

具体内容可以参考《抛出异常来源方式》章节。

 

2) 必须验证异常类型、异常属性、异常原因

 

在验证抛出异常时,必须验证异常类型、异常属性、异常原因等。

 

 

正例:


image.png

 例:

image.png


3) 验证抛出异常后,必须验证相关方法调用

 

在验证抛出异常后,必须验证相关方法调用,来保证单元测试用例走的是期望分支。

 

正例:

 

image.png



《Java单元测试实战》——无效单测:那些年,我们写过的无效单元测试(11) https://developer.aliyun.com/article/1232104?groupCode=java

 

相关文章
|
12天前
|
存储 Java 开发者
Java Map实战:用HashMap和TreeMap轻松解决复杂数据结构问题!
【10月更文挑战第17天】本文深入探讨了Java中HashMap和TreeMap两种Map类型的特性和应用场景。HashMap基于哈希表实现,支持高效的数据操作且允许键值为null;TreeMap基于红黑树实现,支持自然排序或自定义排序,确保元素有序。文章通过具体示例展示了两者的实战应用,帮助开发者根据实际需求选择合适的数据结构,提高开发效率。
42 2
|
17天前
|
存储 消息中间件 安全
JUC组件实战:实现RRPC(Java与硬件通过MQTT的同步通信)
【10月更文挑战第9天】本文介绍了如何利用JUC组件实现Java服务与硬件通过MQTT的同步通信(RRPC)。通过模拟MQTT通信流程,使用`LinkedBlockingQueue`作为消息队列,详细讲解了消息发送、接收及响应的同步处理机制,包括任务超时处理和内存泄漏的预防措施。文中还提供了具体的类设计和方法实现,帮助理解同步通信的内部工作原理。
JUC组件实战:实现RRPC(Java与硬件通过MQTT的同步通信)
|
1天前
|
Java 测试技术 Maven
Java一分钟之-PowerMock:静态方法与私有方法测试
通过本文的详细介绍,您可以使用PowerMock轻松地测试Java代码中的静态方法和私有方法。PowerMock通过扩展Mockito,提供了强大的功能,帮助开发者在复杂的测试场景中保持高效和准确的单元测试。希望本文对您的Java单元测试有所帮助。
8 2
|
9天前
|
Java 程序员 测试技术
Java|让 JUnit4 测试类自动注入 logger 和被测 Service
本文介绍如何通过自定义 IDEA 的 JUnit4 Test Class 模板,实现生成测试类时自动注入 logger 和被测 Service。
17 5
|
27天前
|
Java 流计算
Flink-03 Flink Java 3分钟上手 Stream 给 Flink-02 DataStreamSource Socket写一个测试的工具!
Flink-03 Flink Java 3分钟上手 Stream 给 Flink-02 DataStreamSource Socket写一个测试的工具!
34 1
Flink-03 Flink Java 3分钟上手 Stream 给 Flink-02 DataStreamSource Socket写一个测试的工具!
|
14天前
|
存储 人工智能 Java
将 Spring AI 与 LLM 结合使用以生成 Java 测试
AIDocumentLibraryChat 项目通过 GitHub URL 为指定的 Java 类生成测试代码,支持 granite-code 和 deepseek-coder-v2 模型。项目包括控制器、服务和配置,能处理源代码解析、依赖加载及测试代码生成,旨在评估 LLM 对开发测试的支持能力。
26 1
|
14天前
|
开发框架 Java 程序员
揭开Java反射的神秘面纱:从原理到实战应用!
本文介绍了Java反射的基本概念、原理及应用场景。反射允许程序在运行时动态获取类的信息并操作其属性和方法,广泛应用于开发框架、动态代理和自定义注解等领域。通过反射,可以实现更灵活的代码设计,但也需注意其性能开销。
33 1
|
17天前
|
安全 Java 测试技术
最佳实践:通义灵码生成单元测试,让单测更简单
本文首先讲述了什么是单元测试、单元测试的价值、一个好的单元测试所具备的原则,进而引入如何去编写一个好的单元测试,通义灵码是如何快速生成单元测试的。
|
2月前
|
缓存 负载均衡 Dubbo
Dubbo技术深度解析及其在Java中的实战应用
Dubbo是一款由阿里巴巴开源的高性能、轻量级的Java分布式服务框架,它致力于提供高性能和透明化的RPC远程服务调用方案,以及SOA服务治理方案。
67 6
|
2月前
|
Java
领略Lock接口的风采,通过实战演练,让你迅速掌握这门高深武艺,成为Java多线程领域的武林盟主
领略Lock接口的风采,通过实战演练,让你迅速掌握这门高深武艺,成为Java多线程领域的武林盟主
32 7