zoukankan      html  css  js  c++  java
  • 22.职责链模式

    职责链模式

    一、职责链模式

    1.基本介绍

    1)职责链模式(Chain of Responsibility Pattern),又叫责任链模式,为请求创建了一个接收者对象的链(简单示意图)。这种模式对请求的发送者和接收者进行解耦。

    2)职责链模式通常每个接收者都包含对另一个接收者的引用。如果一个对象不能处理该请求,那么它会把相同的请求传给下一个接收者,依此类推。

    3)这种类型的设计模式属于行为型模式

    2.原理类图

    1)Handler:抽象的处理者,定义了一个处理请求的接口,同时含义另外Handler

    2)ConcreteHandlerA,B是具体的处理者,处理它自己负责的请求,可以访问它的后继者(即下一个处理者),如果可以处理当前请求,则处理,否则就将该请求交个后继者去处理,从而形成一个职责链

    3)Request,含义很多属性,表示一个请求

    二、应用实例

    1.实例要求

    采购员采购教学器材

    • 如果金额小于等于5000,由教学主任审批

    • 如果金额小于等于10000,由院长审批

    • 如果金额小于等于30000,由副校长审批

    • 如果金额超过30000以上,有校长审批

    2.思路分析

    3.代码实现

    public class Client {
    	public static void main(String[] args) {
    		//创建一个请求
    		PurchaseRequest purchaseRequest = new PurchaseRequest(1, 31000, 1);
    		
    		//创建相关的审批人
    		DepartmentApprover departmentApprover = new DepartmentApprover("张主任");
    		CollegeApprover collegeApprover = new CollegeApprover("李院长");
    		ViceSchoolMasterApprover viceSchoolMasterApprover = new ViceSchoolMasterApprover("王副校");
    		SchoolMasterApprover schoolMasterApprover = new SchoolMasterApprover("佟校长");
    	
    		//需要将各个审批级别的下一个设置好 (处理人构成环形: )
    		departmentApprover.setApprover(collegeApprover);
    		collegeApprover.setApprover(viceSchoolMasterApprover);
    		viceSchoolMasterApprover.setApprover(schoolMasterApprover);
    		schoolMasterApprover.setApprover(departmentApprover);
    		
    		departmentApprover.processRequest(purchaseRequest);
    		viceSchoolMasterApprover.processRequest(purchaseRequest);
    	}
    }
    
    public abstract class Approver {
    	Approver approver;  //下一个处理者
    	String name; // 名字
    	
    	public Approver(String name) {
    		this.name = name;
    	}
    
    	//下一个处理者
    	public void setApprover(Approver approver) {
    		this.approver = approver;
    	}
    	
    	//处理审批请求的方法,得到一个请求, 处理是子类完成,因此该方法做成抽象
    	public abstract void processRequest(PurchaseRequest purchaseRequest);
    }
    
    public class DepartmentApprover extends Approver {
    	public DepartmentApprover(String name) {
    		super(name);
    	}
    	
    	@Override
    	public void processRequest(PurchaseRequest purchaseRequest) {
    		if(purchaseRequest.getPrice() <= 5000) {
    			System.out.println(" 请求编号 id= " + purchaseRequest.getId() + " 被 " + this.name + " 处理");
    		}else {
    			approver.processRequest(purchaseRequest);
    		}
    	}
    }
    
    public class CollegeApprover extends Approver {
    	public CollegeApprover(String name) {
    		super(name);
    	}
    	
    	@Override
    	public void processRequest(PurchaseRequest purchaseRequest) {
    		if(purchaseRequest.getPrice() < 5000 && purchaseRequest.getPrice() <= 10000) {
    			System.out.println(" 请求编号 id= " + purchaseRequest.getId() + " 被 " + this.name + " 处理");
    		}else {
    			approver.processRequest(purchaseRequest);
    		}
    	}
    }
    
    public class ViceSchoolMasterApprover extends Approver {
    	public ViceSchoolMasterApprover(String name) {
    		super(name);
    	}
    	
    	@Override
    	public void processRequest(PurchaseRequest purchaseRequest) {
    		if(purchaseRequest.getPrice() < 10000 && purchaseRequest.getPrice() <= 30000) {
    			System.out.println(" 请求编号 id= " + purchaseRequest.getId() + " 被 " + this.name + " 处理");
    		}else {
    			approver.processRequest(purchaseRequest);
    		}
    	}
    }
    
    //请求类
    public class PurchaseRequest {
    	private int type = 0; //请求类型
    	private float price = 0.0f; //请求金额
    	private int id = 0;
    	//构造器
    	public PurchaseRequest(int type, float price, int id) {
    		this.type = type;
    		this.price = price;
    		this.id = id;
    	}
    	public int getType() {
    		return type;
    	}
    	public float getPrice() {
    		return price;
    	}
    	public int getId() {
    		return id;
    	}
    }
    

    三、注意事项

    1)将请求和处理分开,实现解耦提高系统的灵活性。

    2)性能会受到影响,特别是在链比较长的时候,因此需要控制链中最大节点数量,一般通过在Handler中设置一个最大节点数量。

    3)应用场景:有多个对象可以处理同一个请求时。例如:多级请求,拦截器。

  • 相关阅读:
    输入一行文字,找出其中大写字母、小写字母、空格、数字以及其他字符各有多少
    有一字符串,包含n个字符。写一函数,将此字符串中从第m个字符开始的全部字符复制成为另一个字符串
    写一函数,求一个字符串的长度。在main函数中输入字符串,并输出其长度
    有n个人围成一圈,顺序排号。从第1个人开始报数(从1到3报数),凡报到3的人退出圈子,问最后留下的是原来第几号的那位
    有n个整数,使前面各数顺序向后移m个位置,最后m个数变成最前面m个数,见图8.43 写一函数实现以上功能,在主函数中输人n个整数和输出调整后的n个数
    输入10个整数,将其中最小的数与第一个数对换, 把最大的数与最后一个数对换
    LOJ#3271. 「JOISC 2020 Day1」建筑装饰 4 DP+找规律
    LOJ#3160. 「NOI2019」斗主地 打表+拉格朗日插值
    CF1386C Joker 双指针+动态树
    LuoguP2605 [ZJOI2010]基站选址 线段树优化DP
  • 原文地址:https://www.cnblogs.com/chao-zjj/p/11336748.html
Copyright © 2011-2022 走看看