- r - 以节省内存的方式增长 data.frame
- ruby-on-rails - ruby/ruby on rails 内存泄漏检测
- android - 无法解析导入android.support.v7.app
- UNIX 域套接字与共享内存(映射文件)
假设我有一个 UserRepository 结构,它封装了与数据库交互的逻辑。这个结构有一组方法,比如:
还有另一个结构(例如,我们称之为 UserService)依赖于 UserRepository 结构。
为了测试 UserService,我需要模拟 UserRepository 的功能。我知道这样做的唯一方法是为 UserRepository 提供接口(interface)并使 UserService 依赖于它而不是 UserRepository 结构。它将允许创建接口(interface)的模拟实现,并在测试中将其设置为 UserService 的依赖项。
最惯用的方法是什么?
1) 如果 UserService 仅依赖于 1 个 UserRepository 的方法(假设是 findAll)——我是否仍应定义一个包含所有存储库方法的接口(interface),或者最好仅为该方法定义一个单独的接口(interface)并将其用作用户服务?如果是这样,它(接口(interface))的最佳名称是什么?如果另一个结构将依赖于 findAll() 和 findById() 方法,我是否应该再次创建另一个接口(interface)?
2) 为这些接口(interface)存储模拟的最佳位置在哪里?是否可以重复使用它们?或者对于不同结构的测试,我需要重新定义模拟?
附言对我来说,单元测试是项目中非常重要的一部分。我想让它们尽可能可读,避免样板代码并专注于它们的逻辑。因此,在不同的测试文件中为相同的接口(interface)创建多个模拟实现对我来说是一个糟糕的选择,因为它会降低测试代码的可读性。
最佳答案
1) 我同意 elevine 所说的,即只需要该结构所需的方法。例如:您有 UserService
需要 FindByName
和 FindAll
,以及 UserAdminService
需要 FindById
、FindAll
和Save
。在那种情况下,您应该有两个接口(interface):
UserProvider
与 FindByName
和 FindAll
UserAdminProvider
,带有 FindById
、FindAll
和 Save
。这还可以让您控制 UserProvider
,例如你知道它不能调用 Save
,所以它不能修改用户。
您可能只需要一个满足这两个接口(interface)的实际实现。
2) 查看testify/mock和 mockery . Mockery 将在 mocks
子包中为您的接口(interface)生成模拟,每个接口(interface)一个。这确实意味着您不能对两个测试使用相同的模拟结构,但这并不重要,因为代码是生成的。您不会在模拟结构中模拟接口(interface)的行为,而是通过设置期望在测试中进行模拟,例如:
func TestThatYouCantLoginWithNonexistentUser(t *testing.T) {
userRepository := new(mocks.UserRepository)
userService := user.NewService(userRepository)
// if the userService calls UserRepository.FindByName("joe"), it will return nil, since there's no such user.
userRepository.On("FindByName", "joe").Return(nil)
_, err := userService.Login("joe", "password")
// err should not be nil and the message should be "user does not exist"
assert.EqualError(t, err, "user does not exist")
// assert that the expectations were met, i.e. FindByName was called with "joe"
userRepository.AssertExpectations(t)
}
这实际上使测试更容易理解,因为当您调用 FindByName("joe")
时,您不必去检查模拟所做的一些其他文件。
关于unit-testing - Go 中最惯用的方法是测试依赖于具有大量方法的结构的代码?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38660219/
这是代码片段。 请说出这种用小内存存储大数据的算法是什么。 public static void main(String[] args) { long longValue = 21474836
所以我使用 imap 从 gmail 和 outlook 接收电子邮件。 Gmail 像这样编码 =?UTF-8?B?UmU6IM69zq3OvyDOtc68zrHOuc67IG5ldyBlbWFpb
很久以前就学会了 C 代码;想用 Scheme 尝试一些新的和不同的东西。我正在尝试制作一个接受两个参数并返回两者中较大者的过程,例如 (define (larger x y) (if (> x
Azure 恢复服务保管库有两个备份配置选项 - LRS 与 GRS 这是一个有关 Azure 恢复服务保管库的问题。 当其驻留区域发生故障时,如何处理启用异地冗余的恢复服务保管库?如果未为恢复服务启
说,我有以下实体: @Entity public class A { @Id @GeneratedValue private Long id; @Embedded private
我有下一个问题。 我有下一个标准: criteria.add(Restrictions.in("entity.otherEntity", getOtherEntitiesList())); 如果我的
如果这是任何类型的重复,我会提前申请,但我找不到任何可以解决我的具体问题的内容。 这是我的程序: import java.util.Random; public class CarnivalGame{
我目前正在使用golang创建一个聚合管道,在其中使用“$ or”运算符查询文档。 结果是一堆需要分组的未分组文档,这样我就可以进入下一阶段,找到两个数据集之间的交集。 然后将其用于在单独的集合中进行
是否可以在正则表达式中创建 OR 条件。 我正在尝试查找包含此类模式的文件名列表的匹配项 第一个案例 xxxxx-hello.file 或者案例二 xxxx-hello-unasigned.file
该程序只是在用户输入行数时创建菱形的形状,因此它有 6 个 for 循环; 3 个循环创建第一个三角形,3 个循环创建另一个三角形,通过这 2 个三角形和 6 个循环,我们得到了一个菱形,这是整个程序
我有一个像这样的查询字符串 www.google.com?Department=Education & Finance&Department=Health 我有这些 li 标签,它们的查询字符串是这样
我有一个带有静态构造函数的类,我用它来读取 app.config 值。如何使用不同的配置值对类进行单元测试。我正在考虑在不同的应用程序域中运行每个测试,这样我就可以为每个测试执行静态构造函数 - 但我
我正在寻找一个可以容纳多个键的容器,如果我为其中一个键值输入保留值(例如 0),它会被视为“或”搜索。 map, int > myContainer; myContainer.insert(make_
我正在为 Web 应用程序创建数据库,并正在寻找一些建议来对可能具有多种类型的单个实体进行建模,每种类型具有不同的属性。 作为示例,假设我想为“数据源”对象创建一个关系模型。所有数据源都会有一些共享属
(1) =>CREATE TABLE T1(id BIGSERIAL PRIMARY KEY, name TEXT); CREATE TABLE (2) =>INSERT INTO T1 (name)
我不确定在使用别名时如何解决不明确的列引用。 假设有两个表,a 和 b,它们都有一个 name 列。如果我加入这两个表并为结果添加别名,我不知道如何为这两个表引用 name 列。我已经尝试了一些变体,
我的查询是: select * from table where id IN (1,5,4,3,2) 我想要的与这个顺序完全相同,不是从1...5,而是从1,5,4,3,2。我怎样才能做到这一点? 最
我正在使用 C# 代码执行动态生成的 MySQL 查询。抛出异常: CREATE TABLE dump ("@employee_OID" VARCHAR(50)); "{"You have an er
我有日期 2016-03-30T23:59:59.000000+0000。我可以知道它的格式是什么吗?因为如果我使用 yyyy-MM-dd'T'HH:mm:ss.SSS,它会抛出异常 最佳答案 Sim
我有一个示例模式,它的 SQL Fiddle 如下: http://sqlfiddle.com/#!2/6816b/2 这个 fiddle 只是根据 where 子句中的条件查询示例数据库,如下所示:
我是一名优秀的程序员,十分优秀!