Spring设计模式-责任链模式
责任链模式
责任链模式是一种行为设计模式,它允许你创建一系列对象,使每个对象都有机会处理请求。在该模式中,请求沿着对象链传递,直到最后一个责任链对象为止,。
责任链模式的要点
- Handler 接口:定义了处理请求的接口,通常包含一个处理请求的方法。
- ConcreteHandler 具体处理者类:实现了处理请求的方法,并决定是否自行处理请求或将请求传递给下一个处理者。
- Client 客户端:创建责任链,并将请求发送到链中的第一个处理者。
在 Spring 框架中,责任链模式经常被用于实现过滤器、拦截器等功能,例如在 Spring Security 中就有一个基于责任链模式的安全过滤器链。
责任链模式在项目中可以用来做什么?
- 校验用户的权限:token权限、URL权限、是否已经被管理员禁用;
- 校验短信的数据:是否手机号码在黑名单里、参数是否合法、平台的短信功能是否开启
- 支付回调:验签、订单状态更新、发送通知告知用户支付成功、统计订单支付成功数量、金额。
案例
验签、订单状态更新、发送通知告知用户支付成功、统计订单支付成功数量、金额这几个步骤进行拆分,每个步骤都是一个链条,具体代码如下:
定义支付回调上下文类,用于获取各个阶段的结果,即是否处理成功;
// 支付回调上下文类
class PaymentCallbackContext {
// 签名是否有效
private boolean signatureValid;
// 订单状态是否更新
private boolean orderStatusUpdated;
// 通知是否发送
private boolean notificationSent;
public boolean isSignatureValid() {
return signatureValid;
}
public boolean isOrderStatusUpdated() {
return orderStatusUpdated;
}
public boolean isNotificationSent() {
return notificationSent;
}
}定义支付回调接口,后面每一个流程都需要实现这个接口
// 支付回调处理
public interface PaymentCallbackHandler {
// 设置下一个责任链
void setNext(PaymentCallbackHandler handler);
// 方法处理
void handle(PaymentCallbackContext context);
}验签处理,一般通过特定的算法和公钥进行加密对比验证
/**
* 验签处理
*/
public class SignatureValidator implements PaymentCallbackHandler {
private PaymentCallbackHandler nextHandler;
public void setNext(PaymentCallbackHandler handler) {
this.nextHandler = handler;
}
public void handle(PaymentCallbackContext context) {
System.out.println("验证签名...");
// 验签逻辑
context.setSignatureValid(true);
// 传递给下一个处理器
PaymentCallbackHandler nextHandler = new OrderStatusUpdater();
nextHandler.handle(context);
}
}下一个执行链:更新订单状态处理
/**
* 订单状态更新
*/
public class OrderStatusUpdater implements PaymentCallbackHandler {
private PaymentCallbackHandler nextHandler;
public void setNext(PaymentCallbackHandler handler) {
this.nextHandler = handler;
}
public void handle(PaymentCallbackContext context) {
if (context.isSignatureValid()) {
System.out.println("更新订单状态...");
// 更新订单状态逻辑
context.setOrderStatusUpdated(true);
// 传递给下一个处理器
if (nextHandler != null) {
nextHandler.handle(context);
}
} else {
System.out.println("验签未通过,无法更新订单状态。");
}
}
}下一个执行链:发送通知,告知用户支付成功
/**
* 发送通知处理器
*/
public class NotificationSender implements PaymentCallbackHandler {
private PaymentCallbackHandler nextHandler;
public void setNext(PaymentCallbackHandler handler) {
this.nextHandler = handler;
}
public void handle(PaymentCallbackContext context) {
if (context.isOrderStatusUpdated()) {
System.out.println("发送支付成功通知...");
// 发送通知逻辑
context.setNotificationSent(true);
} else {
System.out.println("订单状态未更新,无法发送通知。");
}
}
}下一个执行链:统计订单支付成功的数量和金额
/**
* 实现统计处理器
*/
public class StatisticsCollector implements PaymentCallbackHandler {
private PaymentCallbackHandler nextHandler;
public void setNext(PaymentCallbackHandler handler) {
this.nextHandler = handler;
}
public void handle(PaymentCallbackContext context) {
if (context.isNotificationSent()) {
System.out.println("统计订单支付成功数量和金额...");
// 统计逻辑
} else {
System.out.println("未发送通知,无法统计订单信息。");
}
}
}最后,进行测试,指定各个链条之间的顺序,调用第一个责任链即可,它就会自动执行后续的责任链,代码如下:
public class test {
public static void main(String[] args) {
new test().startPaymentCallback();
}
private void startPaymentCallback() {
// 创建处理器实例
PaymentCallbackHandler signatureValidator = new SignatureValidator();
PaymentCallbackHandler orderStatusUpdater = new OrderStatusUpdater();
PaymentCallbackHandler notificationSender = new NotificationSender();
PaymentCallbackHandler statisticsCollector = new StatisticsCollector();
// 构建责任链
signatureValidator.setNext(orderStatusUpdater);
orderStatusUpdater.setNext(notificationSender);
notificationSender.setNext(statisticsCollector);
// 执行责任链
signatureValidator.handle(new PaymentCallbackContext());
}
}
总结
将支付回调的各个步骤使用责任链模式进行拆分以后,有很多优点:
- 解耦合:你无需关注你的下一个链条是如何实现的,只需要把自己的模块实现好即可,并且链条之间的顺序可以随意切换,在构建责任链时指定Next即可,无需去各个实现类中进行修改代码;
- 高可扩展性:当需要新增加或者修改支付回调处理的步骤时,可以通过添加新的处理器来实现,而无需修改已有的代码。
- 提高可读性:每一个步骤的实现代码都在独立的类中实现,这样可以更容易地理解和调试代码,不像很多业务代码一样,一大堆业务处理逻辑放在一个方法里面调来调去,很容易搞迷糊。
但是他也有一些注意事项,如下:
- 可能造成循环调用: 如果责任链的 setNext() 设置不当,就可能会导致循环调用,栈溢出StackOverFlow;
- 不一定适合所有情况: 责任链模式可以提高系统的灵活性和可扩展性,但是会增加代码的复杂性和代码量。在一些简单的场景下,使用责任链模式可能会显得过于复杂,也没必要;
- 适合场景:整体的业务流程场景较为复杂,并且每个步骤之间都是有序的,需要根据不同的条件选择不同的处理流程时,责任链模式可以提供一种灵活的解决方案。
评论