Chain of Responsibility Design Pattern

 

What is the Chain of Responsibility Design Pattern?

Chain of Responsibility Pattern or Chain of Responsibility Method is a Behavioral Design Pattern, which allows an object to send a request to other objects without knowing who is going to handle it.

  • This pattern is frequently used in the chain of multiple objects, where each object either handles the request or passes it on to the next object in the chain if it is unable to handle that request.
  • This pattern encourages loose coupling between sender and receiver, providing freedom in handling the request.

Characteristics of the Chain of Responsibility Design Pattern

Below are the main characteristics of chain of responsibility design pattern:


  • Loose Coupling: This means the sender of a request doesn’t need to know which specific object will handle it. Similarly, the handler doesn’t need to understand how the requests are sent. This keeps the components separate and flexible.
  • Dynamic Chain: While the program is running, changing the chain is simple. This makes your code incredibly flexible because you may add or delete handlers without changing the main body of the code.
  • Single Responsibility Principle: Each handler in the chain has one job: either to handle the request or to pass it to the next handler. This keeps the code organized and focused, making it easier to manage.
  • Sequential Order: Requests move through the chain one at a time. Each handler gets a chance to process the request in a specific order, ensuring consistency.
  • Fallback Mechanism: If a request isn’t handled by any of the handlers, the chain can include a fallback option. This means there’s a default way to deal with requests that don’t fit anywhere else.
  • RealWorldExampleChainOfResponsibility-(3)

    How to Implement Chain of Responsibiility Design Pattern?

    Below are the main steps for how to implement chain of responsibility design pattern:

    • Step 1: Define the Handler Interface: Create an interface with methods for setting the next handler and processing requests.
    • Step 2: Create Concrete Handlers: Implement the handler interface in multiple classes, each handling specific requests and passing unhandled requests to the next handler.
    • Step 3: Set Up the Chain: Create instances of your handlers and link them together by setting the next handler for each one.
    • Step 4: Send Requests: Use the first handler in the chain to send requests, allowing each handler to decide whether to process it or pass it along.
    • Benefit of Using the Chain of Responsibility in this scenario:

      The Chain of Responsibility pattern is beneficial in this situation because it allows us to create a chain of handlers, where each handler can either handle a request or pass it to the next handler in the chain. This way, we can easily add or remove handlers without modifying the client code, providing flexibility and scalability in handling customer requests.

      chainofresponsibiltydesignpatternclassdiagram-

  • 1. Handler Interface

    Defines the interface for handling requests. Includes methods for handling requests (handleRequest()) and setting the next handler in the chain (setNextHandler()).

    public interface SupportHandler {
        void handleRequest(Request request);
        void setNextHandler(SupportHandler nextHandler);
    }

    2. Concrete Handlers

    Implement the SupportHandler interface. Each handler is responsible for handling requests based on its assigned priority level. If a handler can handle the request, it processes it; otherwise, it passes the request to the next handler in the chain.

    public class Level1SupportHandler implements SupportHandler {
        private SupportHandler nextHandler;
    
        public void setNextHandler(SupportHandler nextHandler) {
            this.nextHandler = nextHandler;
        }
    
        public void handleRequest(Request request) {
            if (request.getPriority() == Priority.BASIC) {
                System.out.println("Level 1 Support handled the request.");
            } else if (nextHandler != null) {
                nextHandler.handleRequest(request);
            }
        }
    }
    
    public class Level2SupportHandler implements SupportHandler {
        private SupportHandler nextHandler;
    
        public void setNextHandler(SupportHandler nextHandler) {
            this.nextHandler = nextHandler;
        }
    
        public void handleRequest(Request request) {
            if (request.getPriority() == Priority.INTERMEDIATE) {
                System.out.println("Level 2 Support handled the request.");
            } else if (nextHandler != null) {
                nextHandler.handleRequest(request);
            }
        }
    }
    
    public class Level3SupportHandler implements SupportHandler {
        public void handleRequest(Request request) {
            if (request.getPriority() == Priority.CRITICAL) {
                System.out.println("Level 3 Support handled the request.");
            } else {
                System.out.println("Request cannot be handled.");
            }
        }
    
        public void setNextHandler(SupportHandler nextHandler) {
            // No next handler for Level 3
        }
    }

  • // Main Class
    public class Main {
        public static void main(String[] args) {
            SupportHandler level1Handler = new Level1SupportHandler();
            SupportHandler level2Handler = new Level2SupportHandler();
            SupportHandler level3Handler = new Level3SupportHandler();
    
            level1Handler.setNextHandler(level2Handler);
            level2Handler.setNextHandler(level3Handler);
    
            Request request1 = new Request(Priority.BASIC);
            Request request2 = new Request(Priority.INTERMEDIATE);
            Request request3 = new Request(Priority.CRITICAL);
    
            level1Handler.handleRequest(request1);
            level1Handler.handleRequest(request2);
            level1Handler.handleRequest(request3);
        }
    }
OUTPUT:
Level 1 Support handled the request.
Level 2 Support handled the request.
Level 3 Support handled the request.


Pros of the Chain of Responsibility Design Pattern

Below are the pros of chain of responsibility design pattern:

  • The pattern makes enables sending a request to a series of possible recipients without having to worry about which object will handle it in the end. This lessens the reliance between items.
  • New handlers can be easily added or existing ones can be modified without affecting the client code. This promotes flexibility and extensibility within the system.
  • The sequence and order of handling requests can be changed dynamically during runtime, which allows adjustment of the processing logic as per the requirements.
  • It simplifies the interaction between the sender and receiver objects, as the sender does not need to know about the processing logic.

Cons of the Chain of Responsibility Design Pattern

Below are the cons of chain of responsibility design pattern:

  • The chain should be implemented correctly otherwise there is a chance that some requests might not get handled at all, which leads to unexpected behavior in the application.
  • The request will go through several handlers in the chain if it is lengthy and complicated, which could cause performance overhead. The processing logic of each handler has an effect on the system’s overall performance.
  • The fact that the chain has several handlers can make debugging more difficult. Tracking the progression of a request and determining which handler is in charge of handling it can be difficult.
  • It may become more difficult to manage and maintain the chain of responsibility if the chain is dynamically modified at runtime.

Comments

Popular posts from this blog

Archunit test

Hexagonal Architecture

visitor design pattern