DEV Community

Hexagonal Architecture in Java

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

  1. Modularity: The separation of concerns makes it easier to manage and modify individual components without affecting the rest of the system.
  2. Testability: Since the core logic is isolated from external dependencies, it becomes easier to write unit tests and mock external systems.
  3. 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:

  1. 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
   }
Enter fullscreen mode Exit fullscreen mode

OrderService.java:

   public interface OrderService {
       void createOrder(Order order);
       Order getOrderById(String id);
   }
Enter fullscreen mode Exit fullscreen mode
  1. 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);
   }
Enter fullscreen mode Exit fullscreen mode
  1. 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();
       }
   }
Enter fullscreen mode Exit fullscreen mode
  1. 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);
       }
   }
Enter fullscreen mode Exit fullscreen mode
  1. 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);
       }
   }
Enter fullscreen mode Exit fullscreen mode

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.

Top comments (0)