- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
这里是 Java 8,但这是一个(很可能)与语言无关的一般单元测试问题。
编写 JUnit 测试的语法很简单,但决定要编写什么测试和如何测试主要/生产代码是我发现最重要的挑战。在阅读单元测试最佳实践时,我一遍又一遍地听到同样的话:
Test the contract
我相信的想法是,单元测试不应该是脆弱的,并且如果方法的实现发生变化,则不一定会中断。该方法应该定义一个输入契约(Contract) -> 结果/结果,并且测试应该旨在验证契约(Contract)是否得到遵守。我想。
假设我有以下方法:
public void doFizzOnBuzz(Buzz buzz, boolean isFoobaz) {
// wsClient is a REST client for a microservice
Widget widget = wsClient.getWidgetByBuzzId(buzz.getId());
if(widget.needsFile()) {
File file = readFileFromFileSystem(buzz.getFile());
if(isFoobaz) {
// Do something with the file (doesn't matter what)
}
}
return;
}
private File readFileFromFileSystem(String filename) {
// Private helper method; implementation doesn't matter here EXCEPT...
// Any checked exceptions that Java might throw (as a result of working)
// with the file system are wrapped in a RuntimeException (hence are now
// unchecked.
// Reads a file from the file system based on the filename/URI you specify
}
所以在这里,我们有一个我们希望为 (doFizzOnBuzz
) 编写单元测试的方法。这个方法:
buzz
和 isFoobaz
wsClient
进行网络/REST 调用readFileFromFileSystem
可能会抛出 RuntimeExceptions
我们可以为此编写什么样的单元测试来“测试合约”?
验证输入(buzz
和 isFoobaz
)是显而易见的;契约(Contract)应定义每个值/状态的有效值/状态,以及它们无效时应发生的异常/结果。
但除此之外,我什至不确定这里的“契约”是什么,这使得为其编写测试非常困难。所以我想这个问题真的应该类似于“我如何确定单元测试的契约是什么,然后你如何编写针对契约而非实现的测试?” p>
但是这个标题对于 SO 问题来说太长了。
最佳答案
您使用 doFizzOnBuzz(Buzz buzz, boolean isFoobaz)
和 private File readFileFromFileSystem(String filename)
方法的代码不容易测试,因为第一种方法将尝试并读取一个文件,这不是你想在测试中做的事情。
在这里,doFizzOnBuzz
需要一些东西来提供一个文件供其使用。这个 FileProvider
(我会这样调用它)可以是一个接口(interface),类似于:
public interface FileProvider {
File getFile(String filename);
}
在生产环境中运行时,会使用实际从磁盘读取文件的实现,但在单元测试 doFizzOnBuzz
时,可以使用 FileProvider
的模拟实现。这将返回一个模拟的 File
。
要记住的关键点是,在测试 doFizzOnBuzz
时,我们不测试提供该文件的任何内容或任何其他内容。我们假设要正常工作。这些其他代码有它们自己的单元测试。
模拟框架,例如 Mockito可用于创建 FileProvider
和 File
的模拟实现,并将模拟 FileProvider
注入(inject)被测类,可能使用 setter:
public void setFileProvider(FileProvider f) {
this.fileProvider = f;
}
此外,我不知道什么是 wsClient
,但我知道它有一个 getWidgetByBuzzId()
方法。这个类也可以是一个接口(interface),出于测试目的,接口(interface)将被模拟,并返回一个模拟的 Widget
,类似于上面的 FileProvider。
使用 mockito,您不仅可以设置接口(interface)的模拟实现,还可以定义在该接口(interface)上调用方法时返回的值:例如
//setup mock FileProvider
FileProvider fp = Mockito.mock(FileProvider.class);
//Setup mock File for FileProvider to return
File mockFile = Mockito.mock(File.class);
Mockito.when(mockFile.getName()).thenReturn("mockfilename");
//other methods...
//Make mock FileProvider return mock File
Mockito.when(fp.getFile("filename")).thenReturn(mockFile);
ClassUnderTest test = new ClassUnderTest();
test.setFileProvider(fp); //inject mock file provider
//Also set up mocks for Buzz,, Widget, and anything else
//run test
test.doFizzOnBuzz(...)
//verify that FileProvider.getFile() was actually called:
Mockito.verify(fp).getFile("filenane");
如果没有使用参数 'filename' 调用 getFile(),则上述测试失败
结论如果您无法直接观察方法的结果,例如它是无效的,你可以使用Mocking来验证它与其他类和方法的交互。
关于java - 如何在返回 void 的方法上进行单元测试 "test the contract"?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36328609/
我正在编写一个具有以下签名的 Java 方法。 void Logger(Method method, Object[] args); 如果一个方法(例如 ABC() )调用此方法 Logger,它应该
我是 Java 新手。 我的问题是我的 Java 程序找不到我试图用作的图像文件一个 JButton。 (目前这段代码什么也没做,因为我只是得到了想要的外观第一的)。这是我的主课 代码: packag
好的,今天我在接受采访,我已经编写 Java 代码多年了。采访中说“Java 垃圾收集是一个棘手的问题,我有几个 friend 一直在努力弄清楚。你在这方面做得怎么样?”。她是想骗我吗?还是我的一生都
我的 friend 给了我一个谜语让我解开。它是这样的: There are 100 people. Each one of them, in his turn, does the following
如果我将使用 Java 5 代码的应用程序编译成字节码,生成的 .class 文件是否能够在 Java 1.4 下运行? 如果后者可以工作并且我正在尝试在我的 Java 1.4 应用程序中使用 Jav
有关于why Java doesn't support unsigned types的问题以及一些关于处理无符号类型的问题。我做了一些搜索,似乎 Scala 也不支持无符号数据类型。限制是Java和S
我只是想知道在一个 java 版本中生成的字节码是否可以在其他 java 版本上运行 最佳答案 通常,字节码无需修改即可在 较新 版本的 Java 上运行。它不会在旧版本上运行,除非您使用特殊参数 (
我有一个关于在命令提示符下执行 java 程序的基本问题。 在某些机器上我们需要指定 -cp 。 (类路径)同时执行java程序 (test为java文件名与.class文件存在于同一目录下) jav
我已经阅读 StackOverflow 有一段时间了,现在我才鼓起勇气提出问题。我今年 20 岁,目前在我的家乡(罗马尼亚克卢日-纳波卡)就读 IT 大学。足以介绍:D。 基本上,我有一家提供簿记应用
我有 public JSONObject parseXML(String xml) { JSONObject jsonObject = XML.toJSONObject(xml); r
我已经在 Java 中实现了带有动态类型的简单解释语言。不幸的是我遇到了以下问题。测试时如下代码: def main() { def ks = Map[[1, 2]].keySet()
一直提示输入 1 到 10 的数字 - 结果应将 st、rd、th 和 nd 添加到数字中。编写一个程序,提示用户输入 1 到 10 之间的任意整数,然后以序数形式显示该整数并附加后缀。 public
我有这个 DownloadFile.java 并按预期下载该文件: import java.io.*; import java.net.URL; public class DownloadFile {
我想在 GUI 上添加延迟。我放置了 2 个 for 循环,然后重新绘制了一个标签,但这 2 个 for 循环一个接一个地执行,并且标签被重新绘制到最后一个。 我能做什么? for(int i=0;
我正在对对象 Student 的列表项进行一些测试,但是我更喜欢在 java 类对象中创建硬编码列表,然后从那里提取数据,而不是连接到数据库并在结果集中选择记录。然而,自从我这样做以来已经很长时间了,
我知道对象创建分为三个部分: 声明 实例化 初始化 classA{} classB extends classA{} classA obj = new classB(1,1); 实例化 它必须使用
我有兴趣使用 GPRS 构建车辆跟踪系统。但是,我有一些问题要问以前做过此操作的人: GPRS 是最好的技术吗?人们意识到任何问题吗? 我计划使用 Java/Java EE - 有更好的技术吗? 如果
我可以通过递归方法反转数组,例如:数组={1,2,3,4,5} 数组结果={5,4,3,2,1}但我的结果是相同的数组,我不知道为什么,请帮助我。 public class Recursion { p
有这样的标准方式吗? 包括 Java源代码-测试代码- Ant 或 Maven联合单元持续集成(可能是巡航控制)ClearCase 版本控制工具部署到应用服务器 最后我希望有一个自动构建和集成环境。
我什至不知道这是否可能,我非常怀疑它是否可能,但如果可以,您能告诉我怎么做吗?我只是想知道如何从打印机打印一些文本。 有什么想法吗? 最佳答案 这里有更简单的事情。 import javax.swin
我是一名优秀的程序员,十分优秀!