Hexagonal Architecture in Java
Adaumir Paixão Victor da Costa
Posted on August 7, 2024
Hexagonal Architecture, also known as Ports and Adapters Architecture, is a design pattern that aims to create loosely coupled application components. This architectural style makes your code more modular, testable, and maintainable by separating the core logic from external concerns like databases, user interfaces, and third-party services.
What is Hexagonal Architecture?
Hexagonal Architecture focuses on creating a core application logic (the hexagon) that is independent of its external interfaces. This core logic interacts with external systems through ports (interfaces) and adapters (implementations of these interfaces). The main idea is to invert the control flow, making the application core completely independent of the outer layers.
Benefits of Hexagonal Architecture
- Modularity: The separation of concerns makes it easier to manage and modify individual components without affecting the rest of the system.
- Testability: Since the core logic is isolated from external dependencies, it becomes easier to write unit tests and mock external systems.
- Maintainability: Clear boundaries between different parts of the system make it easier to understand and maintain the codebase.
Implementing Hexagonal Architecture in Java
Here’s how you can implement Hexagonal Architecture in a Java application:
- Define the Core Domain: Create the core domain classes and interfaces that represent the business logic.
Order.java:
public class Order {
private String id;
private List<OrderItem> items;
// Constructor, getters, and business logic methods
}
OrderService.java:
public interface OrderService {
void createOrder(Order order);
Order getOrderById(String id);
}
- Create the Ports: Define interfaces for interacting with external systems. These are the ports.
OrderRepository.java:
public interface OrderRepository {
void save(Order order);
Order findById(String id);
}
- Implement the Adapters: Create implementations of the ports that interact with external systems. These are the adapters.
OrderRepositoryImpl.java:
public class OrderRepositoryImpl implements OrderRepository {
// Implementation using a database or any other storage mechanism
@Override
public void save(Order order) {
// Save order to the database
}
@Override
public Order findById(String id) {
// Retrieve order from the database
return new Order();
}
}
- Wire Everything Together: Use a dependency injection framework like Spring to wire the core logic, ports, and adapters together.
OrderServiceImpl.java:
@Service
public class OrderServiceImpl implements OrderService {
private final OrderRepository orderRepository;
@Autowired
public OrderServiceImpl(OrderRepository orderRepository) {
this.orderRepository = orderRepository;
}
@Override
public void createOrder(Order order) {
orderRepository.save(order);
}
@Override
public Order getOrderById(String id) {
return orderRepository.findById(id);
}
}
- Controller: Create a controller to handle HTTP requests and delegate them to the service.
OrderController.java:
@RestController
@RequestMapping("/orders")
public class OrderController {
private final OrderService orderService;
@Autowired
public OrderController(OrderService orderService) {
this.orderService = orderService;
}
@PostMapping
public ResponseEntity<Void> createOrder(@RequestBody Order order) {
orderService.createOrder(order);
return ResponseEntity.ok().build();
}
@GetMapping("/{id}")
public ResponseEntity<Order> getOrderById(@PathVariable String id) {
Order order = orderService.getOrderById(id);
return ResponseEntity.ok(order);
}
}
Conclusion
Hexagonal Architecture helps you build modular, testable, and maintainable Java applications by decoupling the core business logic from external systems. By following this architecture, you can ensure that your codebase remains clean and adaptable to change.
Posted on August 7, 2024
Join Our Newsletter. No Spam, Only the good stuff.
Sign up to receive the latest update from our blog.