JUnit测试中的Guice注入器[关闭]

Posted

技术标签:

【中文标题】JUnit测试中的Guice注入器[关闭]【英文标题】:Guice injector in JUnit tests [closed] 【发布时间】:2011-08-03 18:49:34 【问题描述】:

使用 Guice,在每个 JUnit 测试类中获取一个新的注入器是一个好习惯吗,因为每个测试类都应该是独立的?

【问题讨论】:

【参考方案1】:

推荐我最近写的这个框架Guice-Behave。

这很简单,使用两个注解就可以在应用程序的同一上下文中运行测试。

您可以在 Guice 模块中定义您的模拟,这样很容易重用它们。

【讨论】:

【参考方案2】:

您确实应该避免在单元测试中使用 Guice,因为每个测试都应该足够小,以便可以管理手动 DI。通过在单元测试中使用 Guice(或任何 DI),您隐藏了一个警告,即您的类变得太大并承担了太多责任。

为了测试引导程序代码和集成测试,可以为每个测试创建不同的注入器。

【讨论】:

我不同意。使用 Guice,您可以使用 @Inject 并注入没有设置器或构造器的字段。它更具可读性。那么在这种情况下手动依赖应该是什么?我更喜欢使用 Injector 而不是手动反射 API,因为它首先出现在我的脑海中。 我从不直接注入没有设置器的字段。我几乎从不使用 setter 注入。我觉得这两者都很丑陋,并且对所述课程的用户隐藏了课程要求。我尝试只使用 ctor 注入。通过在单元测试中使用 Guice(或任何 DI),你隐藏了一个警告,即你的类变得越来越大并承担了太多的责任。 您是否倾向于编写“浅层”单元测试来模拟测试对象的直接依赖关系?我发现如果您使用真实存储等编写“全栈”测试,手动创建大部分依赖关系树可能会很麻烦。不过,不想争论哪种测试方法更好。 没有“更好”,只有“更适合这个用例”。 JUnit 框架用于运行集成测试时会怎样?【参考方案3】:

这取决于所使用的 JUnit 版本。我们的团队已成功使用 Junit4,现在正在研究 JUnit5。

在 Junit5 中,我们使用扩展。

    public class InjectionPoint implements BeforeTestExecutionCallback 

        @Override
        public void beforeTestExecution(ExtensionContext context) throws Exception 

            List<Module> modules = Lists.newArrayList(new ConfigurationModule());

            Optional<Object> test = context.getTestInstance();

            if (test.isPresent()) 
                RequiresInjection requiresInjection = test.get().getClass().getAnnotation(RequiresInjection.class);

                if (requiresInjection != null) 
                    for (Class c : requiresInjection.values()) 
                        modules.add((Module) c.newInstance());
                    
                

                Module aggregate = Modules.combine(modules);
                Injector injector = Guice.createInjector(aggregate);

                injector.injectMembers(test.get());
                getStore(context).put(injector.getClass(), injector);
            

        

        private Store getStore(ExtensionContext context) 
            return context.getStore(Namespace.create(getClass()));
        

    

然后每个测试使用RequiresInjection注解,它可以接受一个内部模块数组进行聚合,或者没有使用默认值。

    @RequiresInjection
    public class Junit5InjectWithoutModuleTest 

        @Inject
        private TestEnvironment environment;

        @Test
        public void shouldAccessFromOuterModule() 
            assertThat(environment).isNotNull();
        

    

这里是注释:

    @ExtendWith(InjectionPoint.class)
    @Retention(RetentionPolicy.RUNTIME)
    @Target(ElementType.TYPE, ElementType.FIELD, ElementType.PARAMETER, ElementType.METHOD)
    public @interface RequiresInjection 

        Class<? extends Module>[] values() default ;

    

JUnit5 对我来说仍然是新的,所以我可能正在研究模板,但到目前为止,扩展似乎可以解决问题。

对于 JUnit4,我们使用类似的方法,除了注入发生在我们自定义测试运行器的 createTest 方法中,然后每个测试都实现一个具有“getModule”方法的 RequiresInjection 接口。

我可能也应该对 TestNG 大喊一声,因为 Guice 支持是内置的。使用就像这样简单:

@Guice(SomeObjectModule.class)
    public class MyTest 

        @Inject
        SomeObject someObject;    

    

【讨论】:

【参考方案4】:

如果有人偶然发现这个问题并想了解如何从单元测试中获得 Guice 注释,请从如下所示的基类扩展您的测试并调用 injector.injectMembers(this);

public class TestBase 
    protected Injector injector = Guice.createInjector(new AbstractModule() 
        @Override
        protected void configure() 
            bind(HelloService.class);
        
    );

    @Before
    public void setup () 
        injector.injectMembers(this);
    

那么你的测试就可以像这样得到一个注入的HelloService

public class HelloServiceTest extends TestBase 
    @Inject
    HelloService service;

    @Test
    public void testService() throws Exception 
       //Do testing here
    

【讨论】:

您应该注意injectMembers 要测试和需要注入的类,而不仅仅是this 这是测试器类。 应该是HelloServiceTest,而不是HelloServletTest和`HelloService服务;`而不是HelloServlet servlet;?我假设是这样并编辑了您的答案。 TestBase 应该是abstract ?【参考方案5】:

看看Guice Berry。

我现在不建议使用它(文档真的很糟糕),但是看看他们的方法可以让你清楚地知道应该如何在 jUnit 中完成 DI。

【讨论】:

如果你决定使用 GuiceBerry,你可以创建 @Provides 也有 @TestScoped 注释(***.com/a/37979254/345648)(或bind(YourClass.class).in(TestScoped.class);)的函数。这告诉 Guice 每个测试只创建一个实例,而不是 @Singleton,它会使组件跨测试重用,或者没有注释,每次注入时都会创建一个新实例(每个测试可能是多个实例)。跨度> 【参考方案6】:

我发现AtUnit 是对 Guice 的极好补充(它甚至可以处理模拟框架集成)。

这使得单元测试类非常清晰和简洁(永远不会在那里看到Injector),并且在适当的情况下,还可以让您将生产绑定作为单元测试的一部分。

【讨论】:

如果我是对的,AtUnit 源代码库的最后一次提交是在 2008 年。【参考方案7】:

我认为使用DI 会使单元测试代码更简单,我总是使用 DI 进行单元测试和集成测试。

如果没有 DI,一切都很难编码。使用Guice Inject or Spring Autowired。就像我下面的测试代码:

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations = "/application-context.xml")
public class When_inexists_user_disabled 
    @Autowired
    IRegistrationService registrationService;

    private int userId;

    @Before
    public void setUp() 
        Logger.getRootLogger().setLevel(Level.INFO);
        Logger.getLogger("org.springframework").setLevel(Level.WARN);
        BasicConfigurator.configure();

        userId = 999;
    

    @Test(expected=UserNotFoundException.class)
    public void user_should_have_disabled() throws UserNotFoundException 
        registrationService.disable(userId);
    


【讨论】:

我个人认为这更难解决,因为我需要查看应用程序上下文文件以找出 IRegistrationService 正在使用的内容,它是否采用任何模拟或存根以及它们是如何设置的。如果一个测试感觉手动编码太难,那么这表明您可能测试太多,或者您的对象可能需要太多才能开始。

以上是关于JUnit测试中的Guice注入器[关闭]的主要内容,如果未能解决你的问题,请参考以下文章

Junit 5 如何使用 Guice DI

Spring Boot 中的 Junit 测试不注入服务

如何从 Guice 的注入器中检索带注释的实例?

使用 Eclipse Scala IDE 中的 spring-data 注入测试 playframework 2.4

SpringBoot + Junit 踩坑:部分 Service 注入失败

SpringBoot + Junit 踩坑:部分 Service 注入失败