Java中23种面试常考的设计模式之责任链模式(Chain of Responsibility)---行为型模式

简介: Java中23种面试常考的设计模式之责任链模式(Chain of Responsibility)---行为型模式

【Java中23种面试常考的设计模式之责任链模式(Chain of Responsibility)---行为型模式】

知识回顾:
>

之前我们讲过的设计模式在这里呦:
【面试最常见的设计模式之单例模式】
【面试最常见的设计模式之工厂模式】
【Java中23种面试常考的设计模式之备忘录模式(Memento)---行为型模式】
【Java中23种面试常考的设计模式之观察者模式(Observer)---行为型模式】
【Java中23种面试常考的设计模式之模板模式(Template)---行为型模式】
【Java中23种面试常考的设计模式之状态模式(State)---行为型模式】
【Java中23种面试常考的设计模式之策略模式(Strategy)---行为型模式】
【Java中23种面试常考的设计模式之迭代器模式(Iterator)---行为型模式】
【Java中23种面试常考的设计模式之访问者模式(Visitor)---行为型模式】
【Java中23种面试常考的设计模式之中介者模式(Mediator)---行为型模式】
【Java中23种面试常考的设计模式之解释器模式(Interpreter)---行为型模式】
【Java中23种面试常考的设计模式之命令模式(Command)---行为型模式】
接下来我们要进行学习的是:【Java中23种面试常考的设计模式之责任链模式(Chain of Responsibility)---行为型模式】。

责任链模式

  1. 责任链设计模式:为请求创建了一个接收者对象的链。这种模式给予请求的类型,对请求的发送者和接收者进行解耦。
  2. 将能够处理同一类请求的对象连成一条链,所提交的请求沿着链传递,链上的对象逐个判断是否有能力处理该请求,如果能则处理,如果不能则传递给链上的下一个对象处理。

解决的问题

  1. 责任链上的处理者负责处理请求,客户只需要将请求发送到责任链上即可,无须关心请求的处理细节和请求的传递,所以责任链将请求的发送者和请求的处理者解耦了。

生产开发中常用的使用场景

  1. Java中,异常机制就是一种责任链模式。
  2. Servlet开发中,过滤器的链式处理。
  3. 等等。。等等。。

责任链模式优点与缺点

优点

  1. 降低耦合度。它将请求的发送者和接收者解耦。
  2. 简化了对象。使得对象不需要知道链的结构。
  3. 增强给对象指派职责的灵活性。通过改变链内的成员或者调动它们的次序,允许动态地新增或者删除责任。
  4. 增加新的请求处理类很方便。
  5. 由于责任链的创建完全在客户端,因此新增新的具体处理者对原有类库没有任何影响,只需添加新的类,然后在客户端调用时添加即可。符合开闭原则。

缺点

  1. 不能保证请求一定被接收。
  2. 调用的时候可能会造成循环调用。

核心角色

Handler:处理请求的规则
ConcreteHandler: 处理请求的实现类
Client--Main:客户端测试类

UML类图

image.png

实现代码

我们实现这样一个栗子:学校请假的流程

• 如果请假天数小于5天,班主任审批
• 如果请假天数大于等于5天,小于15天,学办负责人审批
• 如果大于等于15天,小于30天,学院书记审批
• 如果大于等于30天,提示拒绝

封装一个对象类

class Request {
   
   

    private String username;

    private int leaveDays;

    private String reason;

    public Request(String username, int leaveDays, String reason) {
   
   
        super();
        this.username = username;
        this.leaveDays = leaveDays;
        this.reason = reason;
    }
    public String getEmpName() {
   
   
        return username;
    }
    public void setEmpName(String empName) {
   
   
        this.username = empName;
    }
    public int getLeaveDays() {
   
   
        return leaveDays;
    }
    public void setLeaveDays(int leaveDays) {
   
   
        this.leaveDays = leaveDays;
    }
    public String getReason() {
   
   
        return reason;
    }
    public void setReason(String reason) {
   
   
        this.reason = reason;
    } 
}

LeaderHandler 类

package com.responsibility;


public abstract class LeaderHandler {
   
   

    protected String name;

    protected LeaderHandler nextLeader; 

    public LeaderHandler(String name) {
   
   
        super();
        this.name = name;
    }

    public void setNextLeader(LeaderHandler nextLeader) {
   
   
        this.nextLeader = nextLeader;
    }

    public abstract void handleRequest(Request request);
}

ConcreteHeadTeacherHandler 类

package com.responsibility;

public class ConcreteHeadTeacherHandler extends LeaderHandler {
   
   

    public ConcreteHeadTeacherHandler(String name) {
   
   
        super(name);
    }

    @Override
    public void handleRequest(Request request) {
   
   
        if(request.getLeaveDays()<5){
   
   
            System.out.println("员工:"+request.getEmpName()+"请假,天数:"+request.getLeaveDays()+",理由:"+request.getReason());
            System.out.println("班主任:"+this.name+",审批通过!");
        }else{
   
   
            if(this.nextLeader!=null){
   
   
                System.out.println("班主任:无法审核,请找相关负责人。。。。");
                this.nextLeader.handleRequest(request);
            }
        }
    }
}

ConcreteDirectorHandler 类

package com.responsibility;

public class ConcreteDirectorHandler extends LeaderHandler {
   
   

    public ConcreteDirectorHandler(String name) {
   
   
        super(name);
    }

    @Override
    public void handleRequest(Request request) {
   
   
        if(request.getLeaveDays()<15){
   
   
            System.out.println("员工:"+request.getEmpName()+"请假,天数:"+request.getLeaveDays()+",理由:"+request.getReason());
            System.out.println("学办负责人:"+this.name+",审批通过!");
        }else{
   
   
            if(this.nextLeader!=null){
   
   
                 System.out.println("学办负责人:无法审核,请找相关负责人。。。。");
                this.nextLeader.handleRequest(request);
            }
        }
    }
}

ConcreteClerkHandler 类

package com.responsibility;

public class ConcreteClerkHandler extends LeaderHandler {
   
   

    public ConcreteClerkHandler(String name) {
   
   
        super(name);
    }

    @Override
    public void handleRequest(Request request) {
   
   
        if(request.getLeaveDays()<30){
   
   
            System.out.println("员工:"+request.getEmpName()+"请假,天数:"+request.getLeaveDays()+",理由:"+request.getReason());
            System.out.println("学院书记:"+this.name+",审批通过!");
        }else{
   
   
            if(this.nextLeader!=null){
   
   
                 System.out.println("请假时间太长啦,不可以哟。。。");
            }
        }
    }
}

客户端测试代码Client-Main

package com.responsibility;

public class Main {
   
   
    public static void main(String[] args) {
   
   
    LeaderHandler a = new ConcreteHeadTeacherHandler("班主任:小红");
    LeaderHandler b = new ConcreteDirectorHandler("学办负责人:小绿");
    LeaderHandler b2 = new ConcreteClerkHandler("学院书记:小紫");

    a.setNextLeader(b);
    b.setNextLeader(b2);

    Request request = new Request("小明", 15, "请假回家!");
    a.handleRequest(request);
}

运行结果展示:
image.png

好了,到这里【Java中23种面试常考的设计模式之责任链模式(Chain of Responsibility)---行为型模式】就结束了,23种设计模式持续更新汇总中。

相关文章
|
21天前
|
消息中间件 缓存 算法
Java多线程面试题总结(上)
进程和线程是操作系统管理程序执行的基本单位,二者有明显区别: 1. **定义与基本单位**:进程是资源分配的基本单位,拥有独立的内存空间;线程是调度和执行的基本单位,共享所属进程的资源。 2. **独立性与资源共享**:进程间相互独立,通信需显式机制;线程共享进程资源,通信更直接快捷。 3. **管理与调度**:进程管理复杂,线程管理更灵活。 4. **并发与并行**:进程并发执行,提高资源利用率;线程不仅并发还能并行执行,提升执行效率。 5. **健壮性**:进程更健壮,一个进程崩溃不影响其他进程;线程崩溃可能导致整个进程崩溃。
26 2
|
21天前
|
存储 Java
Java面向对象面试题总结(上)
在Java中,重写(Override)与重载(Overload)是两个重要的概念,关联到方法的定义与调用。重写是指子类对继承自父类的方法进行新的实现,以便提供子类特有的行为,其关键在于方法签名一致但方法体不同。重载则允许在同一个类中定义多个同名方法,只要参数列表不同即可,以此提供方法调用的灵活性。重写关注多态性,而重载强调编译时多态。
17 1
|
22天前
|
NoSQL Java 数据库
2022年整理最详细的java面试题、掌握这一套八股文、面试基础不成问题[吐血整理、纯手撸]
这篇文章是一份详尽的Java面试题总结,涵盖了从面向对象基础到分布式系统设计的多个知识点,适合用来准备Java技术面试。
2022年整理最详细的java面试题、掌握这一套八股文、面试基础不成问题[吐血整理、纯手撸]
|
11天前
|
C# Windows 开发者
当WPF遇见OpenGL:一场关于如何在Windows Presentation Foundation中融入高性能跨平台图形处理技术的精彩碰撞——详解集成步骤与实战代码示例
【8月更文挑战第31天】本文详细介绍了如何在Windows Presentation Foundation (WPF) 中集成OpenGL,以实现高性能的跨平台图形处理。通过具体示例代码,展示了使用SharpGL库在WPF应用中创建并渲染OpenGL图形的过程,包括开发环境搭建、OpenGL渲染窗口创建及控件集成等关键步骤,帮助开发者更好地理解和应用OpenGL技术。
49 0
|
18天前
|
Java 编译器 开发工具
JDK vs JRE:面试大揭秘,一文让你彻底解锁Java开发和运行的秘密!
【8月更文挑战第24天】JDK(Java Development Kit)与JRE(Java Runtime Environment)是Java环境中两个核心概念。JDK作为开发工具包,不仅包含JRE,还提供编译器等开发工具,支持Java程序的开发与编译;而JRE仅包含运行Java程序所需的组件如JVM和核心类库。一个简单的&quot;Hello, World!&quot;示例展示了两者用途:需借助JDK编译程序,再利用JRE或JDK中的运行环境执行。因此,开发者应基于实际需求选择安装JDK或JRE。
37 0
|
21天前
|
存储 缓存 安全
Java多线程面试题总结(中)
Java内存模型(JMM)定义了程序中所有变量的访问规则与范围,确保多线程环境下的数据一致性。JMM包含主内存与工作内存的概念,通过8种操作管理两者间的交互,确保原子性、可见性和有序性。`synchronized`和`volatile`关键字提供同步机制,前者确保互斥访问,后者保证变量更新的可见性。多线程操作涉及不同状态,如新建(NEW)、可运行(RUNNABLE)等,并可通过中断、等待和通知等机制协调线程活动。`volatile`虽不确保线程安全,但能确保变量更新对所有线程可见。
15 0
|
21天前
|
缓存 安全 Java
Java基础面试题总结(上)
Java有8种基本数据类型:byte(1字节)、short(2字节)、int(4字节)、long(8字节)、float(4字节)、double(8字节)、boolean、char(2字节)。String类被`final`修饰,不可被继承。String为只读,内容不可改;StringBuffer和StringBuilder可修改内容,前者线程安全,后者非线程安全,故效率更高。
13 0
|
22天前
|
算法 Java
【多线程面试题十八】、说一说Java中乐观锁和悲观锁的区别
这篇文章讨论了Java中的乐观锁和悲观锁的区别,其中悲观锁假设最坏情况并在访问数据时上锁,如通过`synchronized`或`Lock`接口实现;而乐观锁则在更新数据时检查是否被其他线程修改,适用于多读场景,并常通过CAS操作实现,如Java并发包`java.util.concurrent`中的类。
|
22天前
|
存储 Java 调度
【多线程面试题 八】、说一说Java同步机制中的wait和notify
Java同步机制中的wait()、notify()、notifyAll()是Object类的方法,用于线程间的通信,其中wait()使当前线程释放锁并进入阻塞状态,notify()唤醒单个等待线程,notifyAll()唤醒所有等待线程。
|
SQL 缓存 安全
Java高频面试题目
面试时面试官最常问的问题总结归纳!
129 0