【问题标题】:Mockito / Junit5 org.opentest4j.AssertionFailedError for getter()Mockito / Junit5 org.opentest4j.AssertionFailedError for getter()
【发布时间】:2023-03-03 10:51:01
【问题描述】:

如果我使用以下方法进行 JUnit Mockito 测试,它会失败并显示 org.opentest4j.AssertionFailedError: expected: not null。但是之前使用 verify() 和 Assertions.assertNotNull() 的 Mockito-(Parameter-)Test 是可以的。我做错了什么或我不明白什么?这是我的代码:

JUnit5/模拟测试:

@Mock
private MyentityAServiceImpl myentityAServiceImplmock;
@Captor
private ArgumentCaptor<MyentityA> myentityAArgument;
@Mock
private MyentityBdao myentityBdaomock;
@Mock
private MyentityB myentityBmock;
@Mock
private Logger loggermock;
@InjectMocks
private MyentityBServiceImpl teServiceImpl;

@Test
public void setMyentityAOfTeIfNullByLanr17() {
    myentityBmock.setLanr7( "1234567" );
    final MyentityA myentityA = new MyentityA();
    mockito.when( myentityAServiceImplmock.findMyentityAByLanr17( myentityBmock.getLanr7() ) ).thenReturn( myentityA );
    mockito.when( myentityBdaomock.save( myentityBmock ) ).thenReturn( myentityBmock );
    myentityBmock = teServiceImpl.setMyentityAOfTeIfNullByLanr17( myentityBmock );
    mockito.verify( myentityBmock ).setMyentityA( myentityAArgument.capture() );
    Assertions.assertNotNull( myentityAArgument );
    Assertions.assertNotNull( myentityBmock.getMyentityA() );  // --> org.opentest4j.AssertionFailedError: expected: not <null>
}

测试方法:

  public MyentityB setMyentityAOfTeIfNullByLanr17( final MyentityB entity ) {
    MyentityA myentityA = entity.getMyentityA();
    if ( myentityA != null ) {
        return entity;
    }
    final String lanr17 = entity.getLanr7();
    myentityA = myentityAServiceImpl.findMyentityAByLanr17( lanr17 );
    if ( myentityA != null ) {
        entity.setMyentityA( myentityA );
        entity.setModuser( "root" );
        return myentityBdao.save( entity );
    }
    return entity;
}

【问题讨论】:

    标签: java mockito junit5


    【解决方案1】:

    抱歉,您的测试代码实在是太复杂了。
    它过于精细地描述了在被测方法中操作的对象的调用流程。
    除了你模拟很多东西:依赖关系,被测方法的参数。
    最后,将被测方法的返回值分配给引用模拟参数的变量。这让事情变得非常不清楚。

    测试必须是直接可以理解的,但事实并非如此。
    我花了大约 500 万来理解错误原因。这么简单的代码很多。

    这个断言失败:

    Assertions.assertNotNull( myentityBmock.getMyentityA() );  // --> org.opentest4j.AssertionFailedError: expected: not <null>
    

    是因为在测试方法中getMynEntityA() 只能是null,因为myentityBmock 是一个模拟,并且像您在此处所做的那样设置一个字段对真实字段和相关的getter 行为没有影响:

    if ( myentityA != null ) {
        entity.setMyentityA( myentityA ); // here you invoke a mocked method.
        entity.setModuser( "root" );
        return myentityBdao.save( entity );
    }
    

    其实你的测试方式主要是描述被测方法的调用流程就够了:

    mockito.verify( myentityBmock ).setMyentityA( myentityAArgument.capture() );
    

    因为您无法测试setMyentityA() 的副作用。

    但实际上我强烈建议您在不模拟测试方法的参数的情况下对您的方法进行单元测试。
    它可能看起来像:

    @Test
    public void setMyentityAOfTeIfNullByLanr17() {
        MyentityB entityB = new MyentityB(...) ;
        entityB.setLanr7( "1234567" );
        final MyentityA myentityA = new MyentityA();
        mockito.when( myentityAServiceImplmock.findMyentityAByLanr17( entityB.getLanr7() ) ).thenReturn( myentityA );
        mockito.when( myentityBdaomock.save(entityB) ).thenReturn(entityB);
    
        // action
        MyentityB entityActualB = teServiceImpl.setMyentityAOfTeIfNullByLanr17(entityB);
    
        // Perform content/logic assertion and no flow assertion :
        Assertions.assertEquals(myEntityA, entityActualB.getMyEntityA());
        Assertions.assertEquals("root", entityActualB.getModuser());
    }
    

    根本没有经过测试的代码,但它应该可以帮助您理解我的意图。

    【讨论】:

    • 现在可以使用了。正如你所说,太多的东西被混合和嘲笑,这使得它太复杂了。我现在明白我的错误了 - 谢谢!
    • 不客气。很高兴您掌握了整体思路。
    • 我明白你的意思 ;-)
    猜你喜欢
    • 1970-01-01
    • 2017-04-19
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多