* translate pattern chain-of-responsability to pt-br * set language pt in README.md --------- Co-authored-by: Ricardo Ramos <0102016812@grupotel.corp>
title, category, language, tag
| title | category | language | tag | |
|---|---|---|---|---|
| Chain of responsibility | Behavioral | pt |
|
Propósito
Evitar acoplar o remetente de uma request ao seu destinatário, dando a mais de um objeto a chance de lidar com a solicitação. Encadeie os objeto recebidos e passe a solicitação ao longo da cadeia até um objeto lidar com isso.
Explicação
Exemplo do mundo real
O Rei Orc dá ordens ao seu exército. O primeiro a receber a ordem é o comandante, então um oficial e depois um soldado. O comandante, o oficial e o soldado formam uma cadeia de responsabilidades.
Em outras palavras
Ajuda a construir uma cadeia de objetos. Uma solicitação entra de uma extremidade e continua indo de um objeto para outro até encontrar um manipulador adequado.
Wikipedia diz
Em orientação a objetos, Chain of Responsibility é um padrão cuja principal função é evitar a dependência entre um objeto receptor e um objeto solicitante. Consiste em uma série de objetos receptores e de objetos de solicitação, onde cada objeto de solicitação possui uma lógica interna que separa quais são tipos de objetos receptores que podem ser manipulados. O restante é passado para os próximos objetos de solicitação da cadeia.
Exemplo de programação
Traduzindo nosso exemplo com os orcs acima. Primeiro, nós temos a classe Request:
public class Request {
private final RequestType requestType;
private final String requestDescription;
private boolean handled;
public Request(final RequestType requestType, final String requestDescription) {
this.requestType = Objects.requireNonNull(requestType);
this.requestDescription = Objects.requireNonNull(requestDescription);
}
public String getRequestDescription() { return requestDescription; }
public RequestType getRequestType() { return requestType; }
public void markHandled() { this.handled = true; }
public boolean isHandled() { return this.handled; }
@Override
public String toString() { return getRequestDescription(); }
}
public enum RequestType {
DEFEND_CASTLE, TORTURE_PRISONER, COLLECT_TAX
}
Em seguida, mostramos a hierarquia no request handler
public interface RequestHandler {
boolean canHandleRequest(Request req);
int getPriority();
void handle(Request req);
String name();
}
@Slf4j
public class OrcCommander implements RequestHandler {
@Override
public boolean canHandleRequest(Request req) {
return req.getRequestType() == RequestType.DEFEND_CASTLE;
}
@Override
public int getPriority() {
return 2;
}
@Override
public void handle(Request req) {
req.markHandled();
LOGGER.info("{} handling request \"{}\"", name(), req);
}
@Override
public String name() {
return "Orc commander";
}
}
// OrcOfficer and OrcSoldier are defined similarly as OrcCommander
O Rei Orc dá as ordens e forma a cadeia
public class OrcKing {
private List<RequestHandler> handlers;
public OrcKing() {
buildChain();
}
private void buildChain() {
handlers = Arrays.asList(new OrcCommander(), new OrcOfficer(), new OrcSoldier());
}
public void makeRequest(Request req) {
handlers
.stream()
.sorted(Comparator.comparing(RequestHandler::getPriority))
.filter(handler -> handler.canHandleRequest(req))
.findFirst()
.ifPresent(handler -> handler.handle(req));
}
}
O chain of responsibility em ação.
var king = new OrcKing();
king.makeRequest(new Request(RequestType.DEFEND_CASTLE, "defend castle"));
king.makeRequest(new Request(RequestType.TORTURE_PRISONER, "torture prisoner"));
king.makeRequest(new Request(RequestType.COLLECT_TAX, "collect tax"));
A saída do console.
Orc commander handling request "defend castle"
Orc officer handling request "torture prisoner"
Orc soldier handling request "collect tax"
Diagrama de classe
Aplicabilidade
Usar Chain of Responsibility quando
- Mais de um objeto pode lidar com uma solicitação, e o manipulador não é conhecido a priori. O manipulador deve ser verificado automaticamente.
- Você deseja emitir uma solicitação para um dos vários objetos sem especificar o destinatário explicitamente.
- O conjunto de objetos que podem lidar com uma solicitação deve ser especificado dinamicamente.
