Mockito框架中@Mock
和@InjectMocks
之间的区别是什么?
答案 0 :(得分:430)
@Mock
创建了一个模拟。 @InjectMocks
创建类的实例,并将使用@Mock
(或@Spy
)注释创建的模拟注入此实例。
请注意,您必须使用@RunWith(MockitoJUnitRunner.class)
或Mockito.initMocks(this)
初始化这些模拟并注入它们。
@RunWith(MockitoJUnitRunner.class)
public class SomeManagerTest {
@InjectMocks
private SomeManager someManager;
@Mock
private SomeDependency someDependency; // this will be injected into someManager
//tests...
}
答案 1 :(得分:125)
这是关于@Mock
和@InjectMocks
如何运作的示例代码。
假设我们有Game
和Player
类。
class Game {
private Player player;
public Game(Player player) {
this.player = player;
}
public String attack() {
return "Player attack with: " + player.getWeapon();
}
}
class Player {
private String weapon;
public Player(String weapon) {
this.weapon = weapon;
}
String getWeapon() {
return weapon;
}
}
如您所见,Game
课程需要Player
来执行attack
。
@RunWith(MockitoJUnitRunner.class)
class GameTest {
@Mock
Player player;
@InjectMocks
Game game;
@Test
public void attackWithSwordTest() throws Exception {
Mockito.when(player.getWeapon()).thenReturn("Sword");
assertEquals("Player attack with: Sword", game.attack());
}
}
Mockito将使用when
和thenReturn
方法模拟播放器类以及它的行为。最后,使用@InjectMocks
Mockito会将Player
放入Game
。
请注意,您甚至不必创建new Game
对象。 Mockito会为你注入它。
// you don't have to do this
Game game = new Game(player);
我们也会使用@Spy
注释获得相同的行为。即使属性名称不同。
@RunWith(MockitoJUnitRunner.class)
public class GameTest {
@Mock Player player;
@Spy List<String> enemies = new ArrayList<>();
@InjectMocks Game game;
@Test public void attackWithSwordTest() throws Exception {
Mockito.when(player.getWeapon()).thenReturn("Sword");
enemies.add("Dragon");
enemies.add("Orc");
assertEquals(2, game.numberOfEnemies());
assertEquals("Player attack with: Sword", game.attack());
}
}
class Game {
private Player player;
private List<String> opponents;
public Game(Player player, List<String> opponents) {
this.player = player;
this.opponents = opponents;
}
public int numberOfEnemies() {
return opponents.size();
}
// ...
那是因为Mockito会检查游戏班的Type Signature
,即Player
和List<String>
。
答案 2 :(得分:69)
在测试类中,测试类应使用@InjectMocks
进行注释。这告诉Mockito哪个类可以注入模拟:
@InjectMocks
private SomeManager someManager;
从那时起,我们可以指定类中的哪些特定方法或对象,在本例中为SomeManager
,将替换为模拟:
@Mock
private SomeDependency someDependency;
在此示例中,SomeDependency
类中的SomeManager
将被模拟。
答案 3 :(得分:48)
@Mock
注释模仿相关对象。
@InjectMocks
注释允许向基础对象注入@Mock
创建的不同(和相关)模拟。
两者都是互补的。
答案 4 :(得分:17)
例如
@Mock
StudentDao studentDao;
@InjectMocks
StudentService service;
@Before
public void setUp() throws Exception {
MockitoAnnotations.initMocks(this);
}
这里我们需要服务类的DAO类。因此,我们模拟它并将其注入服务类实例中。 类似地,在Spring框架中,所有 @Autowired bean都可以通过jUnits中的 @Mock 进行模拟,并通过@InjectMocks注入到bean中。
MockitoAnnotations.initMocks(this)
方法初始化这些模拟并为每个测试方法注入它们,因此需要在setUp()
方法中调用它。
答案 5 :(得分:12)
模拟框架&#34;,Mockito所基于的,是一个框架,使您能够创建Mock对象(旧的术语,这些对象可以被称为分流,因为它们作为dependend的分流器工作功能) 换句话说,模拟对象用于模仿代码依赖的真实对象,您可以使用模拟框架创建代理对象。 通过在测试中使用模拟对象,您实际上是从正常的单元测试到集成测试
Mockito是一个在MIT许可下发布的Java开源测试框架,它是一个&#34;模拟框架&#34;,它允许您使用干净简单的API编写漂亮的测试。 Java空间中有许多不同的模拟框架,但实际上有两种主要类型的模拟对象框架,一种是通过代理实现的,另一种是通过类重映射实现的。
像Spring这样的依赖注入框架允许您在不修改任何代码的情况下注入代理对象,模拟对象需要调用某个方法并返回预期结果。
@InjectMocks
注释尝试实例化测试对象实例,并将带有@Mock
或@Spy
注释的字段注入测试对象的私有字段。
MockitoAnnotations.initMocks(this)
调用,重置测试对象并重新初始化模拟,所以请记住在@Before
/ @BeforeMethod
注释中进行此操作。
答案 6 :(得分:8)
@Tom提到的方法的一个优点是您不必在SomeManager中创建任何构造函数,因此限制客户端实例化它。
@RunWith(MockitoJUnitRunner.class)
public class SomeManagerTest {
@InjectMocks
private SomeManager someManager;
@Mock
private SomeDependency someDependency; // this will be injected into someManager
//You don't need to instantiate the SomeManager with default contructor at all
//SomeManager someManager = new SomeManager();
//Or SomeManager someManager = new SomeManager(someDependency);
//tests...
}
是否良好实践取决于您的应用程序设计。
答案 7 :(得分:5)
尽管已经涵盖了以上答案,但我只是尝试添加一些我看不见的小细节。他们背后的原因(为什么)。
插图:
Sample.java
---------------
public class Sample{
DependencyOne dependencyOne;
DependencyTwo dependencyTwo;
public SampleResponse methodOfSample(){
dependencyOne.methodOne();
dependencyTwo.methodTwo();
...
return sampleResponse;
}
}
SampleTest.java
-----------------------
@RunWith(PowerMockRunner.class)
@PrepareForTest({ClassA.class})
public class SampleTest{
@InjectMocks
Sample sample;
@Mock
DependencyOne dependencyOne;
@Mock
DependencyTwo dependencyTwo;
@Before
public void init() {
MockitoAnnotations.initMocks(this);
}
public void sampleMethod1_Test(){
//Arrange the dependencies
DependencyResponse dependencyOneResponse = Mock(sampleResponse.class);
Mockito.doReturn(dependencyOneResponse).when(dependencyOne).methodOne();
DependencyResponse dependencyTwoResponse = Mock(sampleResponse.class);
Mockito.doReturn(dependencyOneResponse).when(dependencyTwo).methodTwo();
//call the method to be tested
SampleResponse sampleResponse = sample.methodOfSample()
//Assert
<assert the SampleResponse here>
}
}
答案 8 :(得分:3)
许多人在此对@Mock
与@InjectMocks
进行了很好的解释。我喜欢它,但是我认为我们的测试和应用程序应该以不需要使用@InjectMocks
的方式编写。
答案 9 :(得分:1)
请注意,@InjectMocks
将是deprecated
弃用@InjectMocks并安排在Mockito 3/4中删除的计划
,您可以在以下位置关注@avp answer and link:
为什么不应该使用InjectMocks注释自动装配字段
答案 10 :(得分:1)
@Mock
用于声明/模拟从属bean的引用,而@InjectMocks
用于模拟要为其创建测试的bean。
例如:
public class A{
public class B b;
public void doSomething(){
}
}
测试类A
:
public class TestClassA{
@Mocks
public class B b;
@InjectMocks
public class A a;
@Test
public testDoSomething(){
}
}
答案 11 :(得分:1)
@InjectMocks批注可用于将模拟字段自动注入到测试对象中。
在下面的示例中,@ InjectMocks已用于将模拟dataMap注入到dataLibrary中。
@Mock
Map<String, String> dataMap ;
@InjectMocks
DataLibrary dataLibrary = new DataLibrary();
@Test
public void whenUseInjectMocksAnnotation_() {
Mockito.when(dataMap .get("aData")).thenReturn("aMeaning");
assertEquals("aMeaning", dataLibrary .getMeaning("aData"));
}