mirror of
https://github.com/tiennm99/java-design-patterns.git
synced 2026-05-16 06:58:54 +00:00
6cd2d0353a
* update yaml frontmatter format * update abstract document * update abstract factory * use the new pattern template * acyclic visitor seo * adapter seo * ambassador seo * acl seo * aaa seo * async method invocation seo * balking seo * bridge seo * builder seo * business delegate and bytecode seo * caching seo * callback seo * chain seo * update headings * circuit breaker seo * client session + collecting parameter seo * collection pipeline seo * combinator SEO * command seo * cqrs seo * commander seo * component seo * composite seo * composite entity seo * composite view seo * context object seo * converter seo * crtp seo * currying seo * dao seo * data bus seo * data locality seo * data mapper seo * dto seo * decorator seo * delegation seo * di seo * dirty flag seo * domain model seo * double buffer seo * double checked locking seo * double dispatch seo * dynamic proxy seo * event aggregator seo * event-based asynchronous seo * eda seo * event queue seo * event sourcing seo * execute around seo * extension objects seo * facade seo * factory seo * factory kit seo * factory method seo * fanout/fanin seo * feature toggle seo * filterer seo * fluent interface seo * flux seo * flyweight seo * front controller seo * function composition seo * game loop seo * gateway seo * guarded suspension seo * half-sync/half-async seo * health check seo * hexagonal seo * identity map seo * intercepting filter seo * interpreter seo * iterator seo * layers seo * lazy loading seo * leader election seo * leader/followers seo * lockable object seo * rename and add seo for marker interface * master-worker seo * mediator seo * memento seo * metadata mapping seo * microservice aggregator seo * api gw seo * microservices log aggregration seo * mvc seo * mvi seo * mvp seo * mvvm seo * monad seo * monitor seo * monostate seo * multiton seo * mute idiom seo * naked objects & notification seo * null object seo * object mother seo * object pool seo * observer seo * optimistic locking seo * page controller seo * page object seo * parameter object seo * partial response seo * pipeline seo * poison pill seo * presentation model seo * private class data seo * producer-consumer seo * promise seo * property seo * prototype seo * proxy seo * queue-based load leveling seo * reactor seo * registry seo * repository seo * RAII seo * retry seo * role object seo * saga seo * separated interface seo * serialized entity seo * serialized lob seo * servant seo * server session seo * service layer seo * service locator seo * service to worker seo * sharding seo * single table inheritance seo * singleton seo * spatial partition seo * special case seo * specification seo * state seo * step builder seo * strangler seo * strategy seo * subclass sandbox seo * table module seo * template method seo * throttling seo * tolerant reader seo * trampoline seo * transaction script seo * twin seo * type object seo * unit of work seo * update method seo * value object seo * version number seo * virtual proxy seo * visitor seo * seo enhancements * seo improvements * SEO enhancements * SEO improvements * SEO additions * SEO improvements * more SEO improvements * rename hexagonal + SEO improvements * SEO improvements * more SEO stuff * SEO improvements * SEO optimizations * SEO enhancements * enchance SEO * improve SEO * SEO improvements * update headers
164 lines
5.4 KiB
Markdown
164 lines
5.4 KiB
Markdown
---
|
||
title: "Arrange/Act/Assert Pattern in Java: Enhance Testing Clarity and Simplicity"
|
||
shortTitle: Arrange/Act/Assert
|
||
description: "Learn how to use the Arrange/Act/Assert pattern to structure your unit tests in Java. Improve readability and maintainability of your code with clear testing phases."
|
||
category: Testing
|
||
language: en
|
||
tag:
|
||
- Code simplification
|
||
- Isolation
|
||
- Testing
|
||
---
|
||
|
||
## Also known as
|
||
|
||
* Given/When/Then
|
||
|
||
## Intent of Arrange/Act/Assert Design Pattern
|
||
|
||
The Arrange/Act/Assert pattern is essential in unit testing in Java. This testing method structures unit tests clearly by dividing them into three distinct sections: setup (Arrange), execution (Act), and verification (Assert).
|
||
|
||
## Detailed Explanation of Arrange/Act/Assert Pattern with Real-World Examples
|
||
|
||
Real-world example
|
||
|
||
> Imagine you are organizing a small event. To ensure everything runs smoothly, you follow a pattern similar to Arrange/Act/Assert:
|
||
>
|
||
> 1. **Arrange**: You set up the venue, prepare the guest list, arrange seating, and organize the catering.
|
||
> 2. **Act**: You conduct the event according to the plan, welcoming guests, serving food, and following the schedule.
|
||
> 3. **Assert**: After the event, you evaluate its success by checking guest feedback, ensuring all tasks were completed, and reviewing if everything went as planned.
|
||
>
|
||
> This clear separation of preparation, execution, and evaluation helps ensure the event is well-organized and successful, mirroring the structured approach of the Arrange/Act/Assert pattern in software testing.
|
||
|
||
In plain words
|
||
|
||
> Arrange/Act/Assert is a testing pattern that organizes tests into three clear steps for easy maintenance.
|
||
|
||
WikiWikiWeb says
|
||
|
||
> Arrange/Act/Assert is a pattern for arranging and formatting code in UnitTest methods.
|
||
|
||
## Programmatic Example of Arrange/Act/Assert Pattern in Java
|
||
|
||
We need to write comprehensive and clear unit test suite for a class. Using the Arrange/Act/Assert pattern in Java testing ensures clarity.
|
||
|
||
Let's first introduce our `Cash` class to be unit tested.
|
||
|
||
```java
|
||
public class Cash {
|
||
|
||
private int amount;
|
||
|
||
Cash(int amount) {
|
||
this.amount = amount;
|
||
}
|
||
|
||
void plus(int addend) {
|
||
amount += addend;
|
||
}
|
||
|
||
boolean minus(int subtrahend) {
|
||
if (amount >= subtrahend) {
|
||
amount -= subtrahend;
|
||
return true;
|
||
} else {
|
||
return false;
|
||
}
|
||
}
|
||
|
||
int count() {
|
||
return amount;
|
||
}
|
||
}
|
||
```
|
||
|
||
Then we write our unit tests according to Arrange/Act/Assert pattern. Notice the clearly separated steps for each unit test.
|
||
|
||
```java
|
||
class CashAAATest {
|
||
|
||
@Test
|
||
void testPlus() {
|
||
//Arrange
|
||
var cash = new Cash(3);
|
||
//Act
|
||
cash.plus(4);
|
||
//Assert
|
||
assertEquals(7, cash.count());
|
||
}
|
||
|
||
@Test
|
||
void testMinus() {
|
||
//Arrange
|
||
var cash = new Cash(8);
|
||
//Act
|
||
var result = cash.minus(5);
|
||
//Assert
|
||
assertTrue(result);
|
||
assertEquals(3, cash.count());
|
||
}
|
||
|
||
@Test
|
||
void testInsufficientMinus() {
|
||
//Arrange
|
||
var cash = new Cash(1);
|
||
//Act
|
||
var result = cash.minus(6);
|
||
//Assert
|
||
assertFalse(result);
|
||
assertEquals(1, cash.count());
|
||
}
|
||
|
||
@Test
|
||
void testUpdate() {
|
||
//Arrange
|
||
var cash = new Cash(5);
|
||
//Act
|
||
cash.plus(6);
|
||
var result = cash.minus(3);
|
||
//Assert
|
||
assertTrue(result);
|
||
assertEquals(8, cash.count());
|
||
}
|
||
}
|
||
```
|
||
|
||
## When to Use the Arrange/Act/Assert Pattern in Java
|
||
|
||
Use Arrange/Act/Assert pattern when
|
||
|
||
* Unit testing, especially within the context of TDD and BDD
|
||
* Anywhere clarity and structure are needed in test cases
|
||
|
||
## Real-World Applications of Arrange/Act/Assert Pattern in Java
|
||
|
||
* This pattern is particularly useful when practicing TDD and/or BDD methodologies in Java.
|
||
* Utilized in various programming languages and testing frameworks, such as JUnit (Java), NUnit (.NET), and xUnit frameworks.
|
||
|
||
## Benefits and Trade-offs of Arrange/Act/Assert Pattern
|
||
|
||
Benefits:
|
||
|
||
* Improved readability of tests by clearly separating the setup, action, and verification steps.
|
||
* Easier maintenance and understanding of tests, as each test is structured in a predictable way.
|
||
* Facilitates debugging by isolating test failures to specific phases within the test.
|
||
|
||
Trade-offs:
|
||
|
||
* May introduce redundancy in tests, as similar arrangements may be repeated across tests.
|
||
* Some complex tests might not fit neatly into this structure, requiring additional context or setup outside these three phases.
|
||
|
||
## Related Java Design Patterns
|
||
|
||
* [Page Object](https://java-design-patterns.com/patterns/page-object/): A pattern for organizing UI tests that can be used in conjunction with Arrange/Act/Assert.
|
||
|
||
## References and Credits
|
||
|
||
* [The Art of Unit Testing: with examples in C#](https://amzn.to/49IbdwO)
|
||
* [Test Driven Development: By Example](https://amzn.to/3wEwKbF)
|
||
* [Unit Testing Principles, Practices, and Patterns: Effective testing styles, patterns, and reliable automation for unit testing, mocking, and integration testing with examples in C#](https://amzn.to/4ayjpiM)
|
||
* [xUnit Test Patterns: Refactoring Test Code](https://amzn.to/4dHGDpm)
|
||
* [Arrange, Act, Assert: What is AAA Testing?](https://blog.ncrunch.net/post/arrange-act-assert-aaa-testing.aspx)
|
||
* [Bill Wake: 3A – Arrange, Act, Assert (NCrunch)](https://xp123.com/articles/3a-arrange-act-assert/)
|
||
* [GivenWhenThen (Martin Fowler)](https://martinfowler.com/bliki/GivenWhenThen.html)
|