gpt4 book ai didi

java-8 - Mockito:验证方法是使用功能参数调用的

转载 作者:行者123 更新时间:2023-12-04 11:49:06 24 4
gpt4 key购买 nike

我有一个简单的场景,试图验证调用方法时的某些行为(即,使用给定参数调用某个方法,在这种情况下为函数指针)。以下是我的类(class):

@SpringBootApplication
public class Application {

public static void main(String[] args) {
ConfigurableApplicationContext context = SpringApplication.run(Application.class, args);

AppBootStrapper bootStrapper = context.getBean(AppBootStrapper.class);
bootStrapper.start();
}
}

@Component
public class AppBootStrapper {
private NetworkScanner networkScanner;
private PacketConsumer packetConsumer;

public AppBootStrapper(NetworkScanner networkScanner, PacketConsumer packetConsumer) {
this.networkScanner = networkScanner;
this.packetConsumer = packetConsumer;
}

public void start() {
networkScanner.addConsumer(packetConsumer::consumePacket);
networkScanner.startScan();
}
}

@Component
public class NetworkScanner {
private List<Consumer<String>> consumers = new ArrayList<>();

public void startScan(){
Executors.newSingleThreadExecutor().submit(() -> {
while(true) {
// do some scanning and get/parse packets
consumers.forEach(consumer -> consumer.accept("Package Data"));
}
});
}

public void addConsumer(Consumer<String> consumer) {
this.consumers.add(consumer);
}
}

@Component
public class PacketConsumer {
public void consumePacket(String packet) {
System.out.println("Packet received: " + packet);
}
}

@RunWith(JUnit4.class)
public class AppBootStrapperTest {

@Test
public void start() throws Exception {
NetworkScanner networkScanner = mock(NetworkScanner.class);
PacketConsumer packetConsumer = mock(PacketConsumer.class);
AppBootStrapper appBootStrapper = new AppBootStrapper(networkScanner, packetConsumer);

appBootStrapper.start();

verify(networkScanner).addConsumer(packetConsumer::consumePacket);
verify(networkScanner, times(1)).startScan();
}
}

我想通过注册数据包消费者(可能稍后注册其他消费者,但这是强制性的)然后调用 startScan 来验证 bootStrapper 确实做了正确的设置。执行测试用例时收到以下错误消息:
Argument(s) are different! Wanted:
networkScanner bean.addConsumer(
com.spring.starter.AppBootStrapperTest$$Lambda$8/438123546@282308c3
);
-> at com.spring.starter.AppBootStrapperTest.start(AppBootStrapperTest.java:24)
Actual invocation has different arguments:
networkScanner bean.addConsumer(
com.spring.starter.AppBootStrapper$$Lambda$7/920446957@5dda14d0
);
-> at com.spring.starter.AppBootStrapper.start(AppBootStrapper.java:12)

从异常中,显然函数指针不一样。

我以正确的方式接近这个吗?我缺少一些基本的东西吗?我到处玩,让一个消费者注入(inject) PacketConsumer 只是为了看看它是否有所不同,这没关系,但我知道这肯定不是正确的方法。

任何帮助,对此的看法将不胜感激。

最佳答案

Java 没有任何“函数指针”的概念;当你看到:

networkScanner.addConsumer(packetConsumer::consumePacket);

Java 实际编译的是(相当于):
networkScanner.addConsumer(new Consumer<String>() {
@Override void accept(String packet) {
packetConsumer.consumePacket(packet);
}
});

这个匿名内部类恰好被称为 AppBootStrapper$$Lambda$7 .因为它没有(也不应该)定义 equals方法,它永远不会等于编译器在您的测试中生成的匿名内部类,它恰好被称为 AppBootStrapperTest$$Lambda$8 .这与方法体相同的事实无关,并且是从相同的方法引用以相同的方式构建的。

如果您在测试中显式生成消费者并将其保存为 static final Consumer<String>字段,然后您可以在测试中传递该引用并进行比较;此时,引用相等应该成立。这应该可以很好地与 lambda 表达式或方法引用一起使用。

更恰当的测试可能是 verify(packetConsumer, atLeastOnce()).consumePacket(...) ,因为 lambda 的内容是一个实现细节,你真的更关心你的组件如何与其他组件协作。这里的抽象应该在 consumePacket水平,不在 addConsumer等级。

查看 this SO question 上的评论和答案.

关于java-8 - Mockito:验证方法是使用功能参数调用的,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38960539/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com