gpt4 book ai didi

java - @InjectMocks 在 Java 6 和 7 中表现不同

转载 作者:搜寻专家 更新时间:2023-10-30 21:31:12 25 4
gpt4 key购买 nike

通过一个非常简单的 Mockito 运行 JUnit 测试和类,当使用 Java 1.6.0_32 和 Java 1.7.0_04 运行测试时,我看到了不同的输出,并且想了解为什么会发生这种情况。我怀疑正在进行某种类型的删除,但想要一个明确的答案。

这是我的示例代码和有关如何从命令行运行的说明:

FooServiceTest.java

import org.junit.*;
import org.junit.runner.*;
import org.mockito.*;
import org.mockito.runners.MockitoJUnitRunner;
import static org.mockito.Mockito.*;
import java.util.*;

@RunWith(MockitoJUnitRunner.class)
public class FooServiceTest {
@Mock Map<String, String> mockStringString;
@Mock Map<String, Integer> mockStringInteger;

@InjectMocks FooService fooService;

public static void main(String[] args) {
new JUnitCore().run(FooServiceTest.class);
}

@Before
public void setup() {
MockitoAnnotations.initMocks(this);
}

@Test
public void checkInjection() {
when(mockStringString.get("foo")).thenReturn("bar");
fooService.println();
}
}

FooService.java

import java.util.*;

public class FooService {
private Map<String, String> stringString = new HashMap<String, String>();
private Map<String, Integer> stringInteger = new HashMap<String, Integer>();

public void println() {
System.out.println(stringString.get("foo") + " " + stringInteger);
}
}

编译并运行这个例子:

  • 将以上内容保存到文件中
  • 下载并放在同一目录junit.4.10.jarmockito-all-1.9.0.jar
  • 设置 PATH 以包含 JDK
  • 编译javac -cp junit-4.10.jar;mockito-all-1.9.0.jar *.java
  • 运行 java -cp .;junit-4.10.jar;mockito-all-1.9.0.jar FooServiceTest

我相信上面的输出是 null {} 因为 @InjectMocks字段注入(inject)无法正确解析类型,因为它们都是 Map 类型。 这是否正确

现在更改其中一个模拟名称以匹配类中的字段应该允许 Mockito 找到匹配项。例如改变

@Mock Map<String, Integer> mockStringInteger;

@Mock Map<String, Integer> stringInteger;

然后使用 Java 1.6.0_32 编译/运行给出(恕我直言,预期的)输出 bar stringInteger 但使用 1.7.0_04 给出 null stringInteger

这是我运行它的方式(从 Windows 7 中的命令行):

E:\src\mockito-test>set PATH="C:\Program Files (x86)\Java\jdk1.6.0_32\bin"
E:\src\mockito-test>javac -cp junit-4.10.jar;mockito-all-1.9.0.jar *.java
E:\src\mockito-test>java -cp .;junit-4.10.jar;mockito-all-1.9.0.jar FooServiceTest
bar stringInteger
E:\src\mockito-test>set PATH="C:\Program Files (x86)\Java\jdk1.7.0_04\bin"
E:\src\mockito-test>javac -cp junit-4.10.jar;mockito-all-1.9.0.jar *.java
E:\src\mockito-test>java -cp .;junit-4.10.jar;mockito-all-1.9.0.jar FooServiceTest
null stringInteger

最佳答案

I believe the output from above is I null {} because @InjectMocks field injection cannot correctly resolve the types since they are both of type Map. Is this correct?

是的,在这些字段上正确 Mockito 无法清除歧义,因此它会忽略这些歧义字段。

With a very simple Mockito run JUnit test and class I am seeing different output when the test is run with Java 1.6.0_32 and Java 1.7.0_04 and want to understand why this is happening.

实际上,差异在于 JDK 6 和 JDK 7 之间 Arrays.sort 和 Collections.sort() 的不同行为。差异在于新算法应该减少 20% 的交换。这可能是使事情在 JDK6 和 JDK7 下工作的交换操作。

如果您只重命名具有相同类型(或相同删除)的字段中的一个模拟字段,请问您是在“自找麻烦”。当模拟无法按类型区分时,您确实应该将所有模拟字段命名为相应的字段,但 Javadoc 并未明确说明。

顺便说一下,非常感谢您报告这种奇怪的行为,我创建了一个 issue on Mockito ,但是现在我不会真正解决这个问题,而是确保跨 JDK 的行为相同。解决这种情况可能需要在保持兼容性的同时编写新的算法,同时您应该根据测试类的字段命名所有字段模拟。

目前要做的事情可能是通过额外的比较来调整比较器,以在 JDK6 和 JDK7 上强制执行相同的顺序。加上在 Javadoc 中添加一些警告。

编辑:进行两遍可能会解决大多数人的问题。

希望对您有所帮助。感谢您发现问题。


此外,您还需要 MockitoAnnotations.initMocks(this); 或运行器 @RunWith(MockitoJUnitRunner.class),两者都使用不是必需的,可能甚至造成一些问题。 :)

关于java - @InjectMocks 在 Java 6 和 7 中表现不同,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10755890/

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