行为型设计模式—职责链模式

职责链模式:从名字可以拆分为 职责 和 链。即能为请求创建一条由多个处理器组成的链路,每个处理器各自负责自己的职责,相互之间没有耦合,完成自己任务后请求对象即传递到链路的下一个处理器进行处理。

如果在写好的执行函数里加上部分步骤,导致需要增加若干个if-else,因为整个流程耦合在一起,修改了以后我们就得把整个流程全测一遍。。 而职责链就是把步骤解耦为执行链条,从而消除牵一发而动全身的后果。(链表插入优点是遍历到前节点插入即可,不需要像顺序表一样移动之后所有元素)

在一些核心的业务中,应用职责链模式能够让我们无痛地扩展业务流程的步骤

实现责任链模式的对象最起码需要包含如下特性:

  • 成员属性

    • nextHandler: 下一个等待被调用的对象实例
  • 成员方法

    • SetNext: 把下一个对象的实例绑定到当前对象的nextHandler属性上;
    • Do: 当前对象业务逻辑入口,他是每个处理对象实现自己逻辑的地方;
    • Execute: 负责职责链上请求的处理和传递;它会调用当前对象的DonextHandler不为空则调用nextHandler.Do

SetNextExecute 这两个行为是每个 ConcreteHandler 都一样的,所以这两个可以交给抽象处理类型来实现,每个具体处理对象再继承抽象类型,即可减少重复操作。

以病人去医院看病这个处理流程为例提供一个具体示例。看病的具体流程如下:

挂号—>诊室看病—>收费处缴费—>药房拿药

利用责任链模式,实现这个流程中的每个步骤,且相互间不耦合,还支持向流程中增加步骤。

先实现职责链模式里的公共部分—即模式的接口和抽象类

type PatientHandler interface {Execute(*patient) errorSetNext(PatientHandler) PatientHandlerDo(*patient) error
}
// 充当抽象类型,实现公共方法,抽象方法不实现留给实现类自己实现
type Next struct {nextHandler PatientHandler
}func (n *Next) SetNext(handler PatientHandler) PatientHandler {n.nextHandler = handlerreturn handler
}func (n *Next) Execute(patient *patient) (err error) {// 调用不到外部类型的 Do 方法,所以 Next 不能实现 Do 方法if n.nextHandler != nil {if err = n.nextHandler.Do(patient); err != nil {return}return n.nextHandler.Execute(patient)}return
}

即使Next实现了Do方法,go语言中语法也不能达到在父类方法中调用子类方法的效果—即在例子里面用Next 类型的Execute方法调用不到外部实现类型的Do方法。

定义职责链要处理的请求,实现处理逻辑和请求传递的DoExecute方法的参数都是流程中要处理的请求。这里是医院接诊的流程,所以定义一个患者类作为流程的请求。

//流程中的请求类--患者
type patient struct {Name              stringRegistrationDone  boolDoctorCheckUpDone boolMedicineDone      boolPaymentDone       bool
}

按照挂号—>诊室看病—>收费处缴费—>药房拿药这个流程定义四个步骤的处理类,来分别实现每个环节的逻辑。

// Reception 挂号处处理器
type Reception struct {Next
}func (r *Reception) Do(p *patient) (err error) {if p.RegistrationDone {fmt.Println("Patient registration already done")return}fmt.Println("Reception registering patient")p.RegistrationDone = truereturn
}// Clinic 诊室处理器--用于医生给病人看病
type Clinic struct {Next
}func (d *Clinic) Do(p *patient) (err error) {if p.DoctorCheckUpDone {fmt.Println("Doctor checkup already done")return}fmt.Println("Doctor checking patient")p.DoctorCheckUpDone = truereturn
}// Cashier 收费处处理器
type Cashier struct {Next
}func (c *Cashier) Do(p *patient) (err error) {if p.PaymentDone {fmt.Println("Payment Done")return}fmt.Println("Cashier getting money from patient patient")p.PaymentDone = truereturn
}// Pharmacy 药房处理器
type Pharmacy struct {Next
}func (m *Pharmacy) Do (p *patient) (err error) {if p.MedicineDone {fmt.Println("Medicine already given to patient")return}fmt.Println("Pharmacy giving medicine to patient")p.MedicineDone = truereturn
}

Recepiton接诊挂号这个步骤提供的逻辑没有调用到,再定义StartHandler 类型,它不提供处理实现只是作为第一个Handler向下转发请求

// StartHandler 不做操作,作为第一个Handler向下转发请求
type StartHandler struct {Next
}// Do 空Handler的Do
func (h *StartHandler) Do(c *patient) (err error) {// 空Handler 这里什么也不做 只是载体 do nothing...return
}

这是Go 语法限制,公共方法Exeute并不能像面向对象那样先调用this.Do 再调用this.nextHandler.Do

把处理类串起来执行

func main() {patientHealthHandler := StartHandler{}//patient := &patient{Name: "abc"}// 设置病人看病的链路patientHealthHandler.SetNext(&Reception{}).// 挂号SetNext(&Clinic{}). // 诊室看病SetNext(&Cashier{}). // 收费处交钱SetNext(&Pharmacy{}) // 药房拿药//还可以扩展,比如中间加入化验科化验,图像科拍片等等// 执行上面设置好的业务流程if err := patientHealthHandler.Execute(patient); err != nil {// 异常fmt.Println("Fail | Error:" + err.Error())return}// 成功fmt.Println("Success")
}

职责链也可以设置中止条件,针对例子就是在Execute方法里加判断,一旦满足中止后就不再继续往链路的下级节点传递请求。

这也是职责链跟装饰器模式的一个区别,装饰器模式无法在增强实体的过程中停止,只能执行完整个装饰链路。

针对那些可能未来经常会变的核心业务流程,可以在设计初期就考虑使用职责链来实现,减轻未来流程不停迭代时不好扩展的痛点。

职责链模式与模板模式的区别在于,模板模式的业务流程通常是规定不变的,而职责链模式业务是可拓展的

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/626902.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

OLAP引擎也能实现高性能向量检索,据说QPS高于milvus!

更多技术交流、求职机会,欢迎关注字节跳动数据平台微信公众号,回复【1】进入官方交流群 随着 LLM 技术应用及落地,数据库需要提高向量分析以及 AI 支持能力,向量数据库及向量检索等能力“异军突起”,迎来业界持续不断关…

Javacript如何实现继承?

在 JavaScript 中,可以使用原型链和构造函数来实现继承。下面分别介绍两种方式的实现方法: 1. 使用原型链实现继承 javascriptCopy Codefunction Parent(name) {this.name name; }Parent.prototype.getName function() {return this.name; };functio…

深入解析阻塞队列BlockingQueue及源码(超详细)

一、基础概念 1.1 BlockingQueue BlockingQueue 是 java.util.concurrent 包提供的用于解决并发生产者 - 消费者问题的最有用的类。 1.1.1 队列类型: 无限队列 (unbounded queue ) - 几乎可以无限增长 有限队列 ( bounded qu…

代码随想录算法训练营第二十九天| 491.递增子序列、46.全排列 、47.全排列 II

代码随想录算法训练营第二十九天| 491.递增子序列、46.全排列 、47.全排列 II 题目 491.递增子序列 给你一个整数数组 nums ,找出并返回所有该数组中不同的递增子序列,递增子序列中 至少有两个元素 。你可以按 任意顺序 返回答案。 数组中可能含有重…

自创C++题目——风扇

预估难度 简单 题目描述 有一个风扇,它有个旋转叶片,每个旋转叶片的编号是,请输出它旋转后,中心点与地面的直线距离哪个叶片最近,输出此旋转叶片的编号。默认以“”的形式。 当时: 当或时,…

海康visionmaster-VM 嵌入:嵌入用户软件界面的方法

描述 环境:VM4.0.0 VS2015 及以上 现象:将 VM 整体嵌入到客户软件界面中? 解答 将 VM 软件整体嵌入到客户软件中,需要利用 Panel 控件,并且需要先启动 VM 软件,具 体代码如下: C# [DllImport(“…

linux NTP服务器配置

需求场景:局域网内多台服务器时间同步,保持一致 前提条件: 1,各服务器已正确安装NTP服务,查询命令如下: rpm -qa | grep ntp,若回显信息中包含ntp-4.2之类的,表示服务器上存在ntp…

电脑监控软件有哪些,那个好用?

在当今信息化时代,电脑已经成为企业和个人工作、生活中不可或缺的工具。 然而,随着网络的普及和电脑使用频率的增加,也带来了一些安全隐患和管理上的挑战。 为了更好地保护电脑安全、规范员工工作纪律,越来越多的企业和个人开始使…

vue2 与 vue3 实现自定义组件v-model双向数据绑定的方式

前言 有时候我们需要对一个组件绑定自定义 v-model,以更方便地实现双向数据 甚至有时候,我们想要实现绑定多个 “v-model”,也就是多个“双向绑定”,好在 vue 3 已经实现了可使用多个 v-model 例如: 自定义表单输入控件…

OpenEuler校正时间、OpenEuler设置时间

[rootlocalhost ~]# timedatectl set-time "2024-01-16 12:21:15" Failed to set time: Automatic time synchronization is enabled 断网情况下设置OpenEuler的时间,直接执行timedatectl set-time "2024-01-16 12:21:15"是报错的,…

day18 找树左下角的值 路径总和 路径总和Ⅱ 从中序与后序遍历序列构造二叉树 从前序与中序遍历序列构造二叉树

题目1:513 找树左下角的值 题目链接:513 找树左下角的值 题意 找出二叉树的最底层 最左边节点的值 (假设二叉树中至少有1个节点) 最底层节点确保是深度最大的叶子节点,最左边的节点却不一定是左孩子 递归遍历 深度最大的叶子节点最…

docker使用nginx部署vue刷新页面404

docker使用nginx部署vue刷新页面404 从docker内部复制出来的配置文件是这样的,但是刷新页面之后就显示404,关键是我两个前端项目都是用的这一个配置文件,但是只有一个项目出现刷新浏览器显示404的问题,这给我搞懵了!&…

【dayjs】类型“Dayjs”上不存在属性“isSameOrAfter”

dayjs中有一些方法是需要使用插件后才能使用,默认情况下,Day.js只提供核心代码,没有安装插件。 解决方法: import dayjs from dayjs;import isSameOrAfter from dayjs/plugin/isSameOrAfter;dayjs.extend(isSameOrAfter);再次使…

AI嵌入式K210项目(5)-串口通讯

文章目录 前言一、什么是UART?二、K210的UART三、实验过程总结 前言 串口通讯是平时大家进行调试最常用的方法,嵌入式应用通常要求一个简单的并且占用系统资源少的方法来传输数据。通用异步收发传输器 (UART)即可以满足这些要求,它能够灵活地…

odoo16 权限继承修改字段显示2

odoo16 权限继承修改字段显示2 上次文章写道:最近在搭建的一个服装批发中心使用的进销存一体化项目,由于客户文化水平低,不想在发货界面显示 好多无用功能,有些是有用的,有些是他不关心的。占在用户角度考虑,用不到的功能都是垃圾。有他们的道理。在隐藏的过程中,出现了…

2011 年考研数二真题解析

一、选择题 【01】【02】【03】【04】【05】【06】【07】【08】 二、填空题 【09】【10】【11】【12】【13】【14】 三、解答题 【15】【16】【17】【18】【19】【20】【21】【22】【23】

A-B 数对(洛谷)

A-B 数对 题目背景 出题是一件痛苦的事情! 相同的题目看多了也会有审美疲劳,于是我舍弃了大家所熟悉的 AB Problem,改用 A-B 了哈哈! 题目描述 给出一串正整数数列以及一个正整数 C,要求计算出所有满足 A - B C 的数…

运筹说 第90期 | 网络计划-图解评审法

前述章节的网络计划方法主要研究以时间为主要参数的确定型网络模型,其中的概率型网络模型也只讨论工作公式的不确定性,并没有对事项或工作的不确定性进行讨论。由于这类网络模型的建立有严格的规则,大量研究与开发类计划尚无法表达。因从本期…

vue中el-radio无法默认选中

页面上不生效&#xff0c;默认什么都不选中 <el-radio-group v-model"queryParams.videoUrlType"><el-radio :label"1">本地上传</el-radio><el-radio :label"2">外部链接</el-radio> </el-radio-group>da…

污水处理厂能耗分析系统,高效节能保护环境的利器

在城市化发展中&#xff0c;水污染问题也愈发凸显&#xff0c;在污水处理过程中&#xff0c;所产生的能源相对较多&#xff0c;企能源消耗的量就比较大&#xff0c;所以成本就会较高。因此&#xff0c;应对污水处理的具体情况需要进行深入分析与研究&#xff0c;明确具体消耗情…