当前位置: 首页 > 面试题库 >

被测单元:Impl还是接口?

聂翼
2023-03-14
问题内容

假设我有实现它的接口和实现类,并且我想为此编写单元测试。我应该测试什么接口或Impl?

这是一个例子:

public interface HelloInterface {
    public void sayHello();
}


public class HelloInterfaceImpl implements HelloInterface {
    private PrintStream target = System.out;


    @Override
    public void sayHello() {
        target.print("Hello World");

    }

    public void setTarget(PrintStream target){
        this.target = target;
    }
}

因此,我有实现它的HelloInterface和HelloInterfaceImpl。什么是被测单元接口或Impl?

我认为应该是HelloInterface。请考虑以下JUnit测试草图:

public class HelloInterfaceTest {
    private HelloInterface hi;

    @Before
    public void setUp() {
        hi = new HelloInterfaceImpl();
    }

    @Test
    public void testDefaultBehaviourEndsNormally() {
        hi.sayHello();
        // no NullPointerException here
    }

    @Test
    public void testCheckHelloWorld() throws Exception {
        ByteArrayOutputStream out = new ByteArrayOutputStream();
        PrintStream target = new PrintStream(out);
        PrivilegedAccessor.setValue(hi, "target", target);
        //You can use ReflectionTestUtils in place of PrivilegedAccessor
        //really it is DI 
        //((HelloInterfaceImpl)hi).setTarget(target);
        hi.sayHello();
        String result = out.toString();
        assertEquals("Hello World", result);

    }
 }

主线实际上是我注释掉的那条线。

((HelloInterfaceImpl)hi).setTarget(target);

方法setTarget()不是我的公共接口的一部分,所以我不想 意外地
调用它。如果我真的想打电话给我,我应该花点时间考虑一下。例如,它可以帮助我发现我真正想做的是依赖注入。它为我打开了整个新机遇的世界。我可以使用一些现有的依赖注入机制(例如Spring的依赖注入机制),我可以像在代码中实际进行的那样自己模拟它,或者采用完全不同的方法。仔细看看,准备PrintSream并不是那么容易,也许我应该改用模拟对象?

编辑 :我想我应该 始终
专注于界面。从我的角度来看setTarget(),它也不是impl类的“合同”的一部分,它对依赖注入很有帮助。我认为从测试的角度来看,任何Impl类的公共方法都应视为私有方法。但是,这并不意味着我忽略了实现细节。

EDIT-2 在有多个实现\多个接口的情况下,我将测试所有实现,但是当我在setUp()方法中声明变量时,肯定会使用接口。


问题答案:

实现是需要测试的单元。当然,这就是您要实例化的内容以及包含程序/业务逻辑的内容。

如果您有一个关键接口,并且想要确保每个实现都正确地遵守该接口,那么您可以编写一个针对该接口的测试套件,并要求传入一个实例(与任何实现类型无关)。

是的,将Mockito用于PrintStream可能会更容易,并不一定总是像在此特定示例中那样避免使用模拟对象。



 类似资料:
  • 另请参见私有/受保护的方法是否应该在单元测试下? EDIT-2在多个实现\多个接口的情况下,我将测试所有的实现,但是当我在方法中声明变量时,我肯定会使用interface。

  • Spring对MockMvc有2个设置: 独立设置 WebApplication Context安装 一般来说,MockMvc用于哪种测试?单元还是集成?或者两者兼而有之? 使用独立设置(运行在Spring应用程序上下文之外)允许您编写单元测试,而使用WebApplication Context设置您可以编写集成测试,这是对的吗?

  • 问题内容: 我有以下查询: 这两者有什么区别? 所有数据库都支持这两种方法吗? JPA TransactionManager和JTA TransactionManager是否不同? 问题答案: JPA实现可以选择自己管理事务(RESOURCE_LOCAL),或由应用程序服务器的JTA实现管理事务。 在大多数情况下,RESOURCE_LOCAL是可以的。这将使用基本的JDBC级事务。缺点是该事务对于

  • 我目前正在与CDI Unit合作一个项目,我遇到了一个奇怪的问题。我试图在一个简单的项目中重现它: 我有一个使用CdiRunner运行的测试类(如下所述:http://jglue.org/cdi-unit-user-guide/我的测试类注入了被测试的单元:UUD。这个类扩展了一个超级类“ParentTestClass”,它目前是无用的。 测试课。爪哇: 正如我提到的,父类是空的。 ParentT

  • 从Selenium文档来看,WebDriver是一个接口,但在Eclipse中是包在项目浏览器中显示为一个类。此外,如果WebDriver是一个接口,则实现它的类(如ChromeDriver或InternetExplorerDriver)应该定义或。我们在哪里可以看到这些方法的方法定义?