经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » Java相关 » 设计模式 » 查看文章
Java中常用的设计模式之责任链模式详解
来源:jb51  时间:2022/2/28 8:51:24  对本文有异议

优点

1.降低耦合度。它将请求的发送者和接收者解耦。

2.简化了对象。使得对象不需要知道链的结构。

3.增强给对象指派职责的灵活性。通过改变链内的成员或者调动它们的次序,允许动态地新增或者删除责任。

4、增加新的请求处理类很方便。

缺点

1.不能保证请求一定被接收。

2.系统性能将受到一定影响,而且在进行代码调试时不太方便,可能会造成循环调用。

3.可能不容易观察运行时的特征,有碍于除错。

使用场景

1.有多个对象可以处理同一个请求,具体哪个对象处理该请求由运行时刻自动确定。

2.在不明确指定接收者的情况下,向多个对象中的一个提交一个请求。

3.可动态指定一组对象处理请求。

一、实现方式

假设一个场景,学校里,校长的职能大于老师,老师的职能大于学生,基于这样的链路关系,学生处理不了的事情上报给老师,老师处理不了的事情上报给校长。

1、处理抽象类

  1. package com.asurplus.common.handle.style1;
  2. /**
  3. * 处理抽象类
  4. */
  5. public abstract class Handler {
  6. /**
  7. * 下一个处理类
  8. */
  9. protected Handler handler;
  10. public void setHandler(Handler handler) {
  11. this.handler = handler;
  12. }
  13. public Handler getHandler() {
  14. return handler;
  15. }
  16.  
  17. /**
  18. * 处理事件
  19. *
  20. * @param request
  21. */
  22. public abstract void handlerRequest(String request);
  23. }

2、学生处理类

  1. package com.asurplus.common.handle.style1;
  2. import lombok.extern.slf4j.Slf4j;
  3. /**
  4. * 学生处理类
  5. */
  6. @Slf4j
  7. public class StudentHandler extends Handler {
  8. @Override
  9. public void handlerRequest(String request) {
  10. if ("打扫卫生".equals(request)) {
  11. log.info("学生处理中");
  12. } else {
  13. this.handler.handlerRequest(request);
  14. }
  15. }
  16. }

学生能处理“打扫卫生”这件事,如果是其他事件,交给他的下一个元素

3、老师处理类

  1. package com.asurplus.common.handle.style1;
  2. import lombok.extern.slf4j.Slf4j;
  3. /**
  4. * 老师处理类
  5. */
  6. @Slf4j
  7. public class TeacherHandler extends Handler {
  8. @Override
  9. public void handlerRequest(String request) {
  10. if ("批改试卷".equals(request)) {
  11. log.info("老师处理中");
  12. } else {
  13. this.handler.handlerRequest(request);
  14. }
  15. }
  16. }

老师能处理“批改试卷”这件事,如果是其他事件,交给他的下一个元素

4、校长处理类

  1. package com.asurplus.common.handle.style1;
  2. import lombok.extern.slf4j.Slf4j;
  3. /**
  4. * 校长处理类
  5. */
  6. @Slf4j
  7. public class HeadHandler extends Handler {
  8. @Override
  9. public void handlerRequest(String request) {
  10. if ("学籍问题".equals(request)) {
  11. log.info("校长处理中");
  12. } else {
  13. log.error("无法处理该事件");
  14. }
  15. }
  16. }

校长能处理“学籍问题”这件事,如果是其他事件,由于我们的责任链只有三级,都处理不了,只能打印日志了

5、测试

  1. package com.asurplus.common.handle.style1;
  2. /**
  3. * 责任链模式
  4. */
  5. public class TestMain {
  6. public static void main(String[] args) {
  7. // 学生处理器
  8. StudentHandler studentHandler = new StudentHandler();
  9. // 老师处理器
  10. TeacherHandler teacherHandler = new TeacherHandler();
  11. // 校长处理器
  12. HeadHandler headHandler = new HeadHandler();
  13. // 老师的上一级是校长
  14. teacherHandler.setHandler(headHandler);
  15. // 学生的上一级是老师
  16. studentHandler.setHandler(teacherHandler);
  17. // 处理 批改试卷 这件事
  18. studentHandler.handlerRequest("批改试卷");
  19. }
  20. }

输出结果

在这里插入图片描述

可以看出,“批改试卷”这件事,被老师处理了。

二、实现方式

假设一个场景,在我们的电商系统中,当创建一个订单的时候,我们需要去校验很多的数据,我们需要去判断该商品存不存在,库存还有没有,价格对不对,等等校验。

1、订单信息类

  1. package com.asurplus.common.handle.style2;
  2. import lombok.Builder;
  3. import lombok.Data;
  4. /**
  5. * 订单信息
  6. */
  7. @Data
  8. @Builder
  9. public class Order {
  10. // 库存
  11. private int stock;
  12. // 单价
  13. private int price;
  14. }

2、订单校验接口

  1. package com.asurplus.common.handle.style2;
  2. /**
  3. * 校验器接口
  4. *
  5. * @param <T>
  6. */
  7. public interface OrderFilter<T> {
  8. /**
  9. * 业务逻辑
  10. *
  11. * @param t
  12. * @return
  13. */
  14. boolean execute(T t);
  15. }

3、库存校验器

  1. package com.asurplus.common.handle.style2;
  2. import lombok.extern.slf4j.Slf4j;
  3. /**
  4. * 库存校验器
  5. */
  6. @Slf4j
  7. public class OrderStockFilter implements OrderFilter<Order> {
  8. @Override
  9. public boolean execute(Order order) {
  10. if (0 >= order.getStock()) {
  11. log.error("库存不足");
  12. return false;
  13. }
  14. return true;
  15. }
  16. }

4、价格校验器

  1. package com.asurplus.common.handle.style2;
  2. import lombok.extern.slf4j.Slf4j;
  3. /**
  4. * 价格校验器
  5. */
  6. @Slf4j
  7. public class OrderPriceFilter implements OrderFilter<Order> {
  8. @Override
  9. public boolean execute(Order order) {
  10. if (0 > order.getPrice()) {
  11. log.error("价格错误");
  12. return false;
  13. }
  14. return true;
  15. }
  16. }

5、测试

  1. package com.asurplus.common.handle.style2;
  2. import lombok.extern.slf4j.Slf4j;
  3. import java.util.Arrays;
  4. import java.util.List;
  5. /**
  6. * 责任链模式
  7. */
  8. @Slf4j
  9. public class TestMain {
  10. public static void main(String[] args) {
  11. // 建造者模式创建一个订单
  12. Order order = Order.builder().stock(0).price(0).build();
  13. // 库存校验器
  14. OrderStockFilter orderQuantityFilter = new OrderStockFilter();
  15. // 价格校验器
  16. OrderPriceFilter orderPriceFilter = new OrderPriceFilter();
  17. // 组装成一个list
  18. List<OrderFilter<Order>> orderFilters = Arrays.asList(orderQuantityFilter, orderPriceFilter);
  19. boolean res = false;
  20. // 循环校验
  21. for (OrderFilter<Order> item : orderFilters) {
  22. res = item.execute(order);
  23. // 其中任何一项不通过就停止校验
  24. if (!res) {
  25. break;
  26. }
  27. }
  28. if (!res) {
  29. log.error("下单失败");
  30. }
  31. }
  32. }

在这里插入图片描述

被我们的“库存校验器”校验不通过,导致下单失败。

总结

本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注w3xue的更多内容!    

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号