Mockito is a powerful Java framework used for unit testing by creating mock objects. It helps developers isolate components of a system, making testing easier, faster, and more reliable. In this tutorial, we'll cover the basics of using Mockito, including creating mocks, stubbing methods, verifying behavior, and more.
Prerequisites
-
Basic knowledge of Java.
-
Familiarity with unit testing frameworks like JUnit.
-
Maven or Gradle build tools.
Setting Up Mockito
Add Mockito to your project by including the dependency in your build file:
Maven
<dependency>
<groupId>org.mockito</groupId>
<artifactId>mockito-core</artifactId>
<version>4.0.0</version>
<scope>test</scope>
</dependency>
Gradle
testImplementation 'org.mockito:mockito-core:4.0.0'
Key Concepts in Mockito
-
Mock Objects:
-
Mocks are fake objects that mimic the behavior of real objects.
-
-
Spies:
-
Spies are partial mocks that wrap real objects, allowing you to stub specific methods while keeping the real behavior of others.
-
-
Stubbing:
-
Stubbing allows you to define what a mocked method should return when called with specific arguments.
-
-
Verification:
-
Verify that a method was called (or not called) on a mock object with specific arguments.
-
Basic Mockito Examples
1. Creating a Mock Object
import org.mockito.Mockito; import static org.mockito.Mockito.*; class Service { public String fetchData() { return "Real Data"; } } public class MockExample { public static void main(String[] args) { // Create a mock object Service serviceMock = mock(Service.class); // Stub a method when(serviceMock.fetchData()).thenReturn("Mock Data"); // Call the method String result = serviceMock.fetchData(); // Print the result System.out.println(result); // Output: Mock Data } }
2. Verifying Behavior
import static org.mockito.Mockito.*; public class VerifyExample { public static void main(String[] args) { Service serviceMock = mock(Service.class); // Call the mock method serviceMock.fetchData(); // Verify the method was called verify(serviceMock).fetchData(); // Verify no other interactions occurred verifyNoMoreInteractions(serviceMock); } }
3. Stubbing with Argument Matchers
import static org.mockito.Mockito.*;
class Calculator {
public int add(int a, int b) {
return a + b;
}
}
public class ArgumentMatcherExample {
public static void main(String[] args) {
Calculator calculatorMock = mock(Calculator.class);
// Stub with argument matchers
when(calculatorMock.add(anyInt(), eq(10))).thenReturn(20);
// Call the method
int result = calculatorMock.add(5, 10);
// Print the result
System.out.println(result); // Output: 20
// Verify interaction
verify(calculatorMock).add(anyInt(), eq(10));
}
}
Advanced Features
1. Spying on Real Objects
import static org.mockito.Mockito.*; class RealService { public String getData() { return "Real Data"; } } public class SpyExample { public static void main(String[] args) { RealService realService = new RealService(); // Create a spy RealService spyService = spy(realService); // Stub a method when(spyService.getData()).thenReturn("Stubbed Data"); // Call the method String result = spyService.getData(); // Print the result System.out.println(result); // Output: Stubbed Data // Verify interaction verify(spyService).getData(); } }
2. Capturing Arguments
import org.mockito.ArgumentCaptor; import static org.mockito.Mockito.*; public class ArgumentCaptorExample { public static void main(String[] args) { Service serviceMock = mock(Service.class); // Call the mock method serviceMock.fetchData(); // Capture arguments ArgumentCaptor<String> captor = ArgumentCaptor.forClass(String.class); verify(serviceMock).fetchData(); // Print captured argument System.out.println("Method called with captured argument: " + captor.getValue()); } }
Annotations in Mockito
Mockito provides annotations to make tests cleaner and easier to read.
Common Annotations
-
@Mock
: Creates a mock object. -
@Spy
: Creates a spy object. -
@InjectMocks
: Injects mock objects into the class under test. -
@Captor
: Captures arguments passed to methods.
Example Using Annotations
import org.junit.jupiter.api.Test;
import org.mockito.InjectMocks;
import org.mockito.Mock;
import org.mockito.MockitoAnnotations;
import static org.mockito.Mockito.*;
class MyService {
private Dependency dependency;
public MyService(Dependency dependency) {
this.dependency = dependency;
}
public String process() {
return dependency.getData();
}
}
class Dependency {
public String getData() {
return "Real Data";
}
}
public class AnnotationExample {
@Mock
private Dependency dependencyMock;
@InjectMocks
private MyService myService;
@Test
public void testProcess() {
MockitoAnnotations.openMocks(this);
// Stub the dependency method
when(dependencyMock.getData()).thenReturn("Mock Data");
// Call the method
String result = myService.process();
// Assert the result
System.out.println(result); // Output: Mock Data
}
}
Best Practices
-
Mock only what is necessary; avoid over-mocking.
-
Use
@InjectMocks
to reduce boilerplate. -
Avoid mocking value objects or data models.
-
Write tests that are easy to understand and maintain.
-
Use
verifyNoMoreInteractions()
to ensure no unintended interactions.
Conclusion
Mockito is an invaluable tool for unit testing in Java. It simplifies the testing process by providing powerful features for mocking, stubbing, and verifying behavior. With practice, you can use Mockito to write clean, reliable, and maintainable tests for your Java applications.