- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
考虑以下(完全人为的)示例:
public class Length {
private static final int MAX_LENGTH = 10;
private final int length;
public Length(int length) {
if (length > MAX_LENGTH)
throw new IllegalArgumentException("Length too long");
this.length = length;
}
}
我想测试当调用长度大于 MAX_LENGTH
时会抛出异常。有多种方法可以对此进行测试,但都有缺点:
@Test(expected = IllegalArgumentException.class)
public void testMaxLength() {
new Length(11);
}
这复制了测试用例中的常量。如果 MAX_LENGTH
变小,这将不再是边缘情况(尽管显然它应该与单独的情况配对以测试边缘的另一侧)。如果它变大,这将失败并需要手动更改(这可能不是坏事)。
可以通过为 MAX_LENGTH
添加 getter 然后将测试更改为:
new Length(Length.getMaxLength());
这看起来好多了,因为如果常量发生变化,则不需要更改测试。另一方面,它公开了一个常量,否则该常量将是私有(private)的,并且它具有同时测试两种方法的重大缺陷 - 如果两种方法都被破坏,测试可能会给出误报。
另一种方法是根本不使用常量,而是注入(inject)依赖项:
interface MaxLength {
int getMaxLength();
}
public class Length {
public static void setMaxLength(MaxLength maxLength);
}
然后可以将“常量”模拟为测试的一部分(此处使用 Mockito 的示例):
MaxLength mockedLength = mock(MaxLength.class);
when(mokedLength.getMaxLength()).thenReturn(17);
Length.setMaxLength(mockedLength);
new Length(18);
这似乎增加了很多复杂性,但并没有带来多少值(value)(假设没有其他理由注入(inject)依赖项)。
在这个阶段,我更喜欢使用第二种方法来公开常量,而不是在测试中对值进行硬编码。但这对我来说似乎并不理想。有更好的选择吗?还是这些案例缺乏可测试性表明存在设计缺陷?
最佳答案
作为Tim在评论中提到,您的目标是确保您的软件按照规范运行。一个这样的规范可能是最大长度始终为 10,此时就没有必要测试长度为 5 或 15 的世界。
这是要问自己的问题:您希望使用具有不同“常量”值的类的可能性有多大?我在这里引用“常量”是因为如果你以编程方式改变值(value),它根本不是一个常数,是吗? :)
如果您的值永远不会改变,您根本不能使用符号常量,只需直接与 10 进行比较并基于(比方说)0、3、10 和11. 这可能会使您的代码和测试有点难以理解(“10 从哪里来?11 从哪里来?”),并且如果您确实有理由改变数字。不推荐。
如果您的值可能永远不会改变,您可以像您一样使用私有(private)命名常量(即静态最终字段)。然后您的代码将很容易更改,尽管您的测试将无法自动调整代码的方式。
您还可以放宽包私有(private)可见性,这将可用于同一包中的测试。 Javadoc(例如 /** Package-private for testing.*/
)或文档注释(例如 @VisibleForTesting
)可能有助于明确您的意图。如果您的常量值旨在不透明且在您的类之外不可用,例如 URL 模板或身份验证 token ,这是一个不错的选择。
您甚至可以将其设为公共(public)常量,您的类的使用者也可以使用它。 对于您的常量长度示例,public static final 字段可能是最好的,假设您系统的其他部分可能想知道它(例如,用于 UI 验证提示或错误消息)。
如果您的值可能会改变,您可以按实例接受它,如 new Length(10)
或 new Length() .setMaxLength(10)
。 (我认为前者是依赖注入(inject)的一种形式,将常量整数作为依赖。)如果您想在测试中使用不同的值,这也是一个好主意,例如在生产中使用最大长度 2048,但是为了实用起见,针对 10 进行测试。 为了制作一个灵活的长度 validator ,这个选项可能是对静态最终字段的一个很好的升级。
只有当您的值可能在实例的生命周期内发生变化时,我才会费心使用 DI 风格的值提供程序。那时,您可以交互式地查询该值,因此它的行为根本不像常量。对于“长度”,这显然是矫枉过正,但对于“最大允许内存”、“最大同时连接数”或其他一些类似的伪常量来说可能不是。
简而言之,您必须决定需要多少控制权,然后您可以从那里选择最直接的选择;作为“默认值”,您可能希望将其设为可见字段或构造函数参数,因为它们往往在简单性和灵 active 之间取得良好的平衡。
关于java - 依赖常量值的单元测试代码,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/33863890/
我获得了一些源代码示例,我想测试一些功能。不幸的是,我在执行程序时遇到问题: 11:41:31 [linqus@ottsrvafq1 example]$ javac -g test/test.jav
我想测试ggplot生成的两个图是否相同。一种选择是在绘图对象上使用all.equal,但我宁愿进行更艰巨的测试以确保它们相同,这似乎是identical()为我提供的东西。 但是,当我测试使用相同d
我确实使用 JUnit5 执行我的 Maven 测试,其中所有测试类都有 @ExtendWith({ProcessExtension.class}) 注释。如果是这种情况,此扩展必须根据特殊逻辑使测试
在开始使用 Node.js 开发有用的东西之前,您的流程是什么?您是否在 VowJS、Expresso 上创建测试?你使用 Selenium 测试吗?什么时候? 我有兴趣获得一个很好的工作流程来开发我
这个问题已经有答案了: What is a NullPointerException, and how do I fix it? (12 个回答) 已关闭 3 年前。 基于示例here ,我尝试为我的
我正在考虑测试一些 Vue.js 组件,作为 Laravel 应用程序的一部分。所以,我有一个在 Blade 模板中使用并生成 GET 的组件。在 mounted 期间请求生命周期钩子(Hook)。假
考虑以下程序: #include struct Test { int a; }; int main() { Test t=Test(); std::cout<
我目前的立场是:如果我使用 web 测试(在我的例子中可能是通过 VS.NET'08 测试工具和 WatiN)以及代码覆盖率和广泛的数据来彻底测试我的 ASP.NET 应用程序,我应该不需要编写单独的
我正在使用 C#、.NET 4.7 我有 3 个字符串,即。 [test.1, test.10, test.2] 我需要对它们进行排序以获得: test.1 test.2 test.10 我可能会得到
我有一个 ID 为“rv_list”的 RecyclerView。单击任何 RecyclerView 项目时,每个项目内都有一个可见的 id 为“star”的 View 。 我想用 expresso
我正在使用 Jest 和模拟器测试 Firebase 函数,尽管这些测试可能来自竞争条件。所谓 flakey,我的意思是有时它们会通过,有时不会,即使在同一台机器上也是如此。 测试和函数是用 Type
我在测试我与 typeahead.js ( https://github.com/angular-ui/bootstrap/blob/master/src/typeahead/typeahead.js
我正在尝试使用 Teamcity 自动运行测试,但似乎当代理编译项目时,它没有正确完成,因为当我运行运行测试之类的命令时,我收到以下错误: fatal error: 'Pushwoosh/PushNo
这是我第一次玩 cucumber ,还创建了一个测试和 API 的套件。我的问题是在测试 API 时是否需要运行它? 例如我脑子里有这个, 启动 express 服务器作为后台任务 然后当它启动时(我
我有我的主要应用程序项目,然后是我的测试的第二个项目。将所有类型的测试存储在该测试项目中是一种好的做法,还是应该将一些测试驻留在主应用程序项目中? 我应该在我的主项目中保留 POJO JUnit(测试
我正在努力弄清楚如何实现这个计数。模型是用户、测试、等级 用户 has_many 测试,测试 has_many 成绩。 每个等级都有一个计算分数(strong_pass、pass、fail、stron
我正在尝试测试一些涉及 OkHttp3 的下载代码,但不幸失败了。目标:测试 下载图像文件并验证其是否有效。平台:安卓。此代码可在生产环境中运行,但测试代码没有任何意义。 产品代码 class Fil
当我想为 iOS 运行 UI 测试时,我收到以下消息: SetUp : System.Exception : Unable to determine simulator version for X 堆
我正在使用 Firebase Remote Config 在 iOS 上设置 A/B 测试。 一切都已设置完毕,我正在 iOS 应用程序中读取服务器端默认值。 但是在多个模拟器上尝试,它们都读取了默认
[已编辑]:我已经用 promise 方式更改了我的代码。 我正在写 React with this starter 由 facebook 创建,我是测试方面的新手。 现在我有一个关于图像的组件,它有
我是一名优秀的程序员,十分优秀!