Unit Testing and Mocking in Java with JUnit 5 and Mockito
Jul 29, 2025 am 01:20 AMUse JUnit 5 and Mockito to effectively isolate dependencies for unit testing. 1. Create mock objects through @Mock, @InjectMocks inject the tested instance, @ExtendWith enables Mockito extension; 2. Use when().thenReturn() to define simulation behavior, verify() to verify the number of method calls and parameters; 3. Can simulate exception scenarios and verify error handling; 4. Recommend constructor injection, avoid over-simulation, and maintain test atomicity; 5. Use assertAll() to merge assertions, and @Nested organizes test scenarios to improve test maintainability and reliability.
Unit testing and mocking are important means to ensure code quality in Java, and JUnit 5 and Mockito are the most popular combination of testing frameworks. They work together to efficiently isolate business logic, especially for scenarios that rely on external services, databases, or complex objects.

Why use JUnit 5 Mockito?
JUnit 5 is a mainstream unit testing framework in Java, providing more flexible annotation models, extension mechanisms and assertion functions. Mockito is a powerful mocking framework that allows you to create and configure "fake" objects to replace real dependencies, thus focusing on testing the behavior of target classes.
Combining the two will allow you to:

- Isolate the tested code from external dependencies
- Verify the number of method calls and parameters
- Simulate exceptions and boundary conditions
- Improve test maintainability and execution speed
1. Basic environment construction
First make sure that the project introduces JUnit 5 and Mockito dependencies. If you are using Maven, add the following to pom.xml
:
<dependencies> <!-- JUnit 5 --> <dependency> <groupId>org.junit.jupiter</groupId> <artifactId>junit-jupiter</artifactId> <version>5.10.0</version> <scope>test</scope> </dependency> <!-- Mockito --> <dependency> <groupId>org.mockito</groupId> <artifactId>mockito-core</artifactId> <version>5.7.0</version> <scope>test</scope> </dependency> <!-- Mockito JUnit Jupiter support (optional, used for automatic initialization of @Mock annotation) --> <dependency> <groupId>org.mockito</groupId> <artifactId>mockito-junit-jupiter</artifactId> <version>5.7.0</version> <scope>test</scope> </dependency> </dependencies>
Also, make sure to use a build configuration that supports JUnit 5 (such as Surefire plugin version >= 2.22.0).

2. Use Mockito to simulate dependency objects
Suppose we have an order service OrderService
which relies on a payment gateway PaymentGateway
interface:
public interface PaymentGateway { boolean charge(double amount); } public class OrderService { private final PaymentGateway paymentGateway; public OrderService(PaymentGateway paymentGateway) { this.paymentGateway = paymentGateway; } public boolean processOrder(double amount) { if (amount <= 0) return false; return paymentGateway.charge(amount); } }
We want to test OrderService.processOrder()
but don't want to actually call the payment system. At this time, you can use Mockito to create a simulated PaymentGateway
.
Write test classes
import org.junit.jupiter.api.Test; import org.mockito.InjectMocks; import org.mockito.Mock; import static org.junit.jupiter.api.Assertions.*; import static org.mockito.Mockito.*; class OrderServiceTest { @Mock private PaymentGateway paymentGateway; @InjectMocks private OrderService orderService; // Must cooperate with MockitoExtension to automatically initialize @Mock and @InjectMocks @org.junit.jupiter.api.extension.ExtendWith(MockitoExtension.class) static class Nested {} @Test void shouldReturnFalse_WhenAmountIsInvalid() { // Call the method under test boolean result = orderService.processOrder(-100); // Assert result assertFalse(result); // Verify that the payment gateway is not called verify(paymentGateway, never()).charge(anyDouble()); } @Test void shouldReturnTrue_WhenPaymentSucceeds() { // Simulate dependency behavior when(paymentGateway.charge(100.0)).thenReturn(true); // Execute boolean result = orderService.processOrder(100.0); // Assert assertTrue(result); verify(paymentGateway).charge(100.0); } @Test void shouldReturnFalse_WhenPaymentFails() { when(paymentGateway.charge(100.0)).thenReturn(false); boolean result = orderService.processOrder(100.0); assertFalse(result); verify(paymentGateway).charge(100.0); } }
Key points description:
-
@Mock
: Create a mock object (mock) -
@InjectMocks
: Create a target class instance and automatically inject the@Mock
tagged fields (through constructor or field injection) -
@ExtendWith(MockitoExtension.class)
: Enable the JUnit extension of Mockito to automatically initialize annotations -
when(...).thenReturn(...)
: Define the return value of the simulation method -
verify(...)
: Verify whether a method has been called, as well as the number of calls, parameters, etc.
3. Behavior Verification
In addition to asserting the return value, you can also verify that the method is called correctly. For example:
@Test void shouldAttemptChargeExactlyOnce() { when(paymentGateway.charge(50.0)).thenReturn(true); orderService.processOrder(50.0); // Verify that the charge method is called once and the parameter is 50.0 verify(paymentGateway, times(1)).charge(eq(50.0)); }
Other common verification methods:
-
never()
: Never called -
atLeastOnce()
: at least once -
timeout(100)
: Called within a specified time (for asynchronous) -
inOrder(...)
: Verify the call order
// Example: Verify the call order InOrder inOrder = inOrder(paymentGateway); inOrder.verify(paymentGateway).charge(30.0); inOrder.verify(paymentGateway).charge(20.0);
4. Handle exceptions and boundary situations
You can have the mock object throw exceptions and test the error handling logic:
@Test void shouldHandlePaymentFailureDueToException() { when(paymentGateway.charge(100.0)).thenThrow(new RuntimeException("Network error")); assertThrows(RuntimeException.class, () -> { orderService.processOrder(100.0); }); verify(paymentGateway).charge(100.0); }
You can also use doThrow()
to cooperate with doReturn()
to deal with void
methods or special scenarios.
5. Tips and best practices
- Avoid oversimulation : only simulate dependencies that really need isolation, and do not simulate the entire link
- Preferred constructor injection : easy to pass mock objects during testing
- Maintain test atomicity : A test only validates one behavior
- Batch assertions using
assertAll()
@Test void multipleAssertions() { when(paymentGateway.charge(100.0)).thenReturn(true); boolean result = orderService.processOrder(100.0); assertAll( () -> assertTrue(result), () -> verify(paymentGateway).charge(100.0) ); }
- Use the
@Nested
test class to organize different scenarios:
@Nested class WhenAmountIsPositive { @Test void shouldChargePaymentGateway() { ... } }
Basically that's it. The combination of JUnit 5 Mockito makes Java unit testing intuitive and powerful. The key is to understand "test behavior" rather than "test implementation" and control dependencies through mock to ensure that the test is fast, stable and repeatable.
The above is the detailed content of Unit Testing and Mocking in Java with JUnit 5 and Mockito. For more information, please follow other related articles on the PHP Chinese website!

Hot AI Tools

Undress AI Tool
Undress images for free

Undresser.AI Undress
AI-powered app for creating realistic nude photos

AI Clothes Remover
Online AI tool for removing clothes from photos.

Clothoff.io
AI clothes remover

Video Face Swap
Swap faces in any video effortlessly with our completely free AI face swap tool!

Hot Article

Hot Tools

Notepad++7.3.1
Easy-to-use and free code editor

SublimeText3 Chinese version
Chinese version, very easy to use

Zend Studio 13.0.1
Powerful PHP integrated development environment

Dreamweaver CS6
Visual web development tools

SublimeText3 Mac version
God-level code editing software (SublimeText3)

Hot Topics

Maven is a standard tool for Java project management and construction. The answer lies in the fact that it uses pom.xml to standardize project structure, dependency management, construction lifecycle automation and plug-in extensions; 1. Use pom.xml to define groupId, artifactId, version and dependencies; 2. Master core commands such as mvnclean, compile, test, package, install and deploy; 3. Use dependencyManagement and exclusions to manage dependency versions and conflicts; 4. Organize large applications through multi-module project structure and are managed uniformly by the parent POM; 5.

SetupaMaven/GradleprojectwithJAX-RSdependencieslikeJersey;2.CreateaRESTresourceusingannotationssuchas@Pathand@GET;3.ConfiguretheapplicationviaApplicationsubclassorweb.xml;4.AddJacksonforJSONbindingbyincludingjersey-media-json-jackson;5.DeploytoaJakar

@property decorator is used to convert methods into properties to implement the reading, setting and deletion control of properties. 1. Basic usage: define read-only attributes through @property, such as area calculated based on radius and accessed directly; 2. Advanced usage: use @name.setter and @name.deleter to implement attribute assignment verification and deletion operations; 3. Practical application: perform data verification in setters, such as BankAccount to ensure that the balance is not negative; 4. Naming specification: internal variables are prefixed, property method names are consistent with attributes, and unified access control is used to improve code security and maintainability.

First, use JavaScript to obtain the user system preferences and locally stored theme settings, and initialize the page theme; 1. The HTML structure contains a button to trigger topic switching; 2. CSS uses: root to define bright theme variables, .dark-mode class defines dark theme variables, and applies these variables through var(); 3. JavaScript detects prefers-color-scheme and reads localStorage to determine the initial theme; 4. Switch the dark-mode class on the html element when clicking the button, and saves the current state to localStorage; 5. All color changes are accompanied by 0.3 seconds transition animation to enhance the user

Yes, a common CSS drop-down menu can be implemented through pure HTML and CSS without JavaScript. 1. Use nested ul and li to build a menu structure; 2. Use the:hover pseudo-class to control the display and hiding of pull-down content; 3. Set position:relative for parent li, and the submenu is positioned using position:absolute; 4. The submenu defaults to display:none, which becomes display:block when hovered; 5. Multi-level pull-down can be achieved through nesting, combined with transition, and add fade-in animations, and adapted to mobile terminals with media queries. The entire solution is simple and does not require JavaScript support, which is suitable for large

To generate hash values using Java, it can be implemented through the MessageDigest class. 1. Get an instance of the specified algorithm, such as MD5 or SHA-256; 2. Call the .update() method to pass in the data to be encrypted; 3. Call the .digest() method to obtain a hash byte array; 4. Convert the byte array into a hexadecimal string for reading; for inputs such as large files, read in chunks and call .update() multiple times; it is recommended to use SHA-256 instead of MD5 or SHA-1 to ensure security.

Use datetime.strptime() to convert date strings into datetime object. 1. Basic usage: parse "2023-10-05" as datetime object through "%Y-%m-%d"; 2. Supports multiple formats such as "%m/%d/%Y" to parse American dates, "%d/%m/%Y" to parse British dates, "%b%d,%Y%I:%M%p" to parse time with AM/PM; 3. Use dateutil.parser.parse() to automatically infer unknown formats; 4. Use .d

Full screen layout can be achieved using Flexbox or Grid. The core is to make the minimum height of the page the viewport height (min-height:100vh); 2. Use flex:1 or grid-template-rows:auto1frauto to make the content area occupy the remaining space; 3. Set box-sizing:border-box to ensure that the margin does not exceed the container; 4. Optimize the mobile experience with responsive media query; this solution is compatible with good structure and is suitable for login pages, dashboards and other scenarios, and finally realizes a full screen page layout with vertical centering and full viewport.
