当前位置: 首页 > news >正文

设计模式第六讲:责任链模式和迭代器模式详解

一. 责任链模式

1. 背景

在现实生活中,常常会出现这样的事例:一个请求有多个对象可以处理,但每个对象的处理条件或权限不同。例如,公司员工请假,可批假的领导有部门负责人、副总经理、总经理等,但每个领导能批准的天数不同,员工必须根据自己要请假的天数去找不同的领导签名,也就是说员工必须记住每个领导的姓名、电话和地址等信息,这增加了难度。这样的例子还有很多,如找领导出差报销、生活中的“击鼓传花”游戏等。

2. 定义和特点

(1). 定义:为了避免请求发送者与多个请求处理者耦合在一起,将所有请求的处理者通过前一对象记住其下一个对象的引用而连成一条链;当有请求发生时,可将请求沿着这条链传递,直到有对象处理它为止。

PS:责任链模式也叫职责链模式。在责任链模式中,客户只需要将请求发送到责任链上即可,无须关心请求的处理细节和请求的传递过程,所以责任链将请求的发送者和请求的处理者解耦了。

(2). 优点:

 A. 降低了对象之间的耦合度。该模式使得一个对象无须知道到底是哪一个对象处理其请求以及链的结构,发送者和接收者也无须拥有对方的明确信息。

 B. 增强了系统的可扩展性。可以根据需要增加新的请求处理类,满足开闭原则。

 C. 增强了给对象指派职责的灵活性。当工作流程发生变化,可以动态地改变链内的成员或者调动它们的次序,也可动态地新增或者删除责任。

 D. 责任链简化了对象之间的连接。每个对象只需保持一个指向其后继者的引用,不需保持其他所有处理者的引用,这避免了使用众多的 if 或者 if···else 语句。

 E. 责任分担。每个类只需要处理自己该处理的工作,不该处理的传递给下一个对象完成,明确各类的责任范围,符合类的单一职责原则。

(3). 缺点:

 A. 不能保证每个请求一定被处理。由于一个请求没有明确的接收者,所以不能保证它一定会被处理,该请求可能一直传到链的末端都得不到处理。

 B. 对比较长的职责链,请求的处理可能涉及多个处理对象,系统性能将受到一定影响。

 C. 职责链建立的合理性要靠客户端来保证,增加了客户端的复杂性,可能会由于职责链的错误设置而导致系统出错,如可能会造成循环调用。

3. 具体实现

(1). 模式结构

 A. 抽象处理者:定义一个处理请求的接口,包含抽象处理方法和一个后继连接。

 B. 具体处理者:实现抽象处理者的处理方法,判断能否处理本次请求,如果可以处理请求则处理,否则将该请求转给它的后继者。

 C. 客户端调用:创建处理链,并向链头的具体处理者对象提交请求,它不关心处理细节和请求的传递过程。

结构图如下:

(2). 使用场景

 员工请假,不同的天数需要被不同级别的领导批准,这个时候可以采用责任链模式。每个领导都是一个具体的请假处理者,同时提炼出来一个抽象处理者,包括:请假处理方法和下一个节点,最后由客户端建立责任链,并将请求传递进去。

更多C++后台开发技术点知识内容包括C/C++,Linux,Nginx,ZeroMQ,MySQL,Redis,MongoDB,ZK,流媒体,音视频开发,Linux内核,TCP/IP,协程,DPDK多个高级知识点。

C/C++Linux服务器开发高级架构师/C++后台开发架构师免费学习地址

【文章福利】另外还整理一些C++后台开发架构师 相关学习资料,面试题,教学视频,以及学习路线图,免费分享有需要的可以点击领取

(3). 代码实操

请假请求类:

    /// <summary>/// 请假请求类/// </summary>public class LeaveRequest{/// <summary>/// 姓名/// </summary>public string name { get; set; }/// <summary>/// 请假天数/// </summary>public int leaveDays { get; set; }}

抽象处理者和具体处理者:

    /// <summary>/// 抽象处理者角色/// </summary>public abstract class AbstractHandler{/// <summary>/// 代表责任链中的下一个角色/// </summary>public AbstractHandler NextHandler { get; set; }/// <summary>/// 处理请假请求/// </summary>public abstract void HandleRequest(LeaveRequest request);}/// <summary>/// 直接领导(一级领导)/// </summary>public class LeaderHandler1 : AbstractHandler{/// <summary>/// 处理请假请求/// </summary>public override void HandleRequest(LeaveRequest request){if (request.leaveDays<=3){Console.WriteLine($"直接领导(一级领导)已经批准{request.name}的请假请求,请假天数:{request.leaveDays}");}else{if (this.NextHandler!=null){this.NextHandler.HandleRequest(request);}else{Console.WriteLine($"{request.name}请假天数太多,没有人批准该假条!");}}}}/// <summary>/// 二级领导/// </summary>public class LeaderHandler2 : AbstractHandler{/// <summary>/// 处理请假请求/// </summary>public override void HandleRequest(LeaveRequest request){if (request.leaveDays > 3 && request.leaveDays <= 7){Console.WriteLine($"二级领导已经批准{request.name}的请假请求,请假天数:{request.leaveDays}");}else{if (this.NextHandler != null){this.NextHandler.HandleRequest(request);}else{Console.WriteLine($"{request.name}请假天数太多,没有人批准该假条!");}}}}/// <summary>/// 最高领导(三级)/// </summary>public class LeaderHandler3 : AbstractHandler{/// <summary>/// 处理请假请求/// </summary>public override void HandleRequest(LeaveRequest request){if (request.leaveDays > 7 && request.leaveDays <= 30){Console.WriteLine($"最高领导(三级)已经批准{request.name}的请假请求,请假天数:{request.leaveDays}");}else{if (this.NextHandler != null){this.NextHandler.HandleRequest(request);}else{Console.WriteLine($"{request.name}请假天数太多,没有人批准该假条!");}}}}

客户单创建责任链并调用:

                //1. 有人要请假LeaveRequest myRequest1 = new LeaveRequest() { name="小明",leaveDays=6};LeaveRequest myRequest2 = new LeaveRequest(){name = "小王",leaveDays = 2};LeaveRequest myRequest3 = new LeaveRequest(){name = "小放",leaveDays = 15};LeaveRequest myRequest4 = new LeaveRequest(){name = "张三",leaveDays = 35};//2. 请假相关的责任链LeaderHandler1 handler1 = new LeaderHandler1();LeaderHandler2 handler2 = new LeaderHandler2();LeaderHandler3 handler3 = new LeaderHandler3();handler1.NextHandler = handler2;handler2.NextHandler = handler3;//3.进行请假处理handler1.HandleRequest(myRequest1);handler1.HandleRequest(myRequest2);handler1.HandleRequest(myRequest3);handler1.HandleRequest(myRequest4);

测试结果:

4. 适用场景分析

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

 (2). 可动态指定一组对象处理请求,或添加新的处理者。

 (3). 在不明确指定请求处理者的情况下,向多个处理者中的一个提交请求。

二. 迭代器模式

1. 背景

  在现实生活以及程序设计中,经常要访问一个聚合对象中的各个元素,如【数据结构】中的链表遍历,通常的做法是将链表的创建和遍历都放在同一个类中,但这种方式不利于程序的扩展,如果要更换遍历方法就必须修改程序源代码,这违背了 “开闭原则”。既然将遍历方法封装在聚合类中不可取,那么聚合类中不提供遍历方法,将遍历方法由用户自己实现是否可行呢?答案是同样不可取,因为这种方式会存在两个缺点:

 A. 暴露了聚合类的内部表示,使其数据不安全;

 B. 增加了客户的负担。

 “迭代器模式”能较好地克服以上缺点,它在客户访问类与聚合类之间插入一个迭代器,这分离了聚合对象与其遍历行为,对客户也隐藏了其内部细节,且满足“单一职责原则”和“开闭原则”。比如C#中的List就实现了迭代器模式。

2. 定义和特点

(1). 定义:提供一个对象顺序访问聚合对象中的一系列数据,而不暴露聚合对象的内部表示。

(2). 优点:

 A. 访问一个聚合对象的内容而无须暴露它的内部表示。

 B. 遍历任务交由迭代器完成,这简化了聚合类。

 C. 它支持以不同方式遍历一个聚合,甚至可以自定义迭代器的子类以支持新的遍历。

 D. 增加新的聚合类和迭代器类都很方便,无须修改原有代码。

 E. 封装性良好,为遍历不同的聚合结构提供一个统一的接口。

(3). 缺点:

 增加了类的个数,这在一定程度上增加了系统的复杂性

3. 具体实现

(1). 模式结构

 迭代器模式是通过将聚合对象的遍历行为分离出来,抽象成迭代器类来实现的,其目的是在不暴露聚合对象的内部结构的情况下,让外部代码透明地访问聚合的内部数据。

 A. 抽象聚合(Aggregate)角色:定义存储、添加、删除聚合对象以及创建迭代器对象的接口。

 B. 具体聚合(ConcreteAggregate)角色:实现抽象聚合类,返回一个具体迭代器的实例。

 C. 抽象迭代器(Iterator)角色:定义访问和遍历聚合元素的接口,通常包含 hasNext()、first()、next() 等方法。

 D. 具体迭代器(Concretelterator)角色:实现抽象迭代器接口中所定义的方法,完成对聚合对象的遍历,记录遍历的当前位置。

结构如下图:

(2). 使用场景

  见下面代码模拟。

(3). 代码实操

聚合接口和具体的聚合类

    /// <summary>/// 抽象聚合接口/// </summary>public interface IAggregate<T>{public void Add(T obj);public void Remove(T obj);/// <summary>/// 获取聚合对象的迭代器/// </summary>/// <returns></returns>public Iterator<T> GetIterator();}/// <summary>/// 具体聚合/// </summary>public class ConcreteAggregate<T> : IAggregate<T>{private List<T> list = new List<T>();public void Add(T obj){list.Add(obj);}public void Remove(T obj){list.Remove(obj);}/// <summary>/// 获取迭代器/// </summary>/// <returns></returns>public Iterator<T> GetIterator(){return new ConcreteIterator<T>(list);}}

抽象迭代器和具体的迭代器

   /// <summary>/// 抽象迭代器/// </summary>public interface Iterator<T>{T First();T Next();bool HasNext();}/// <summary>/// 具体的迭代器/// </summary>/// <typeparam name="T"></typeparam>public class ConcreteIterator<T> : Iterator<T>{private List<T> list = null;private int index = 0;public ConcreteIterator(List<T> myList){this.list = myList;}public T First(){return list[0];}public bool HasNext(){if (index<list.Count-1){return true;}else{return false;}}public T Next(){T obj =default(T);if (this.HasNext()){obj = list[++index];}return obj;}}

测试

{//聚合对象添加数据IAggregate<string> ag = new ConcreteAggregate<string>();ag.Add("001");ag.Add("002");ag.Add("003");ag.Add("004");//获取对应的迭代器Iterator<string> it = ag.GetIterator();//输出数据Console.WriteLine($"第一个元素为:{it.First()}");Console.WriteLine("剩下的元素为:");while (it.HasNext()){Console.WriteLine(it.Next());}
}

运行结果

4. 适用场景分析

 (1). 当需要为聚合对象提供多种遍历方式时。

 (2). 当需要为遍历不同的聚合结构提供一个统一的接口时。

 (3). 当访问一个聚合对象的内容而无须暴露其内部细节的表示时。

PS: 由于聚合与迭代器的关系非常密切,所以大多数语言在实现聚合类时都提供了迭代器类,因此大数情况下使用语言中已有的聚合类的迭代器就已经够了。

原文链接:https://www.cnblogs.com/yaopengfei/p/13489597.html

相关文章:

设计模式第六讲:责任链模式和迭代器模式详解

一. 责任链模式1. 背景在现实生活中&#xff0c;常常会出现这样的事例&#xff1a;一个请求有多个对象可以处理&#xff0c;但每个对象的处理条件或权限不同。例如&#xff0c;公司员工请假&#xff0c;可批假的领导有部门负责人、副总经理、总经理等&#xff0c;但每个领导能批…...

K8s 架构简介(一)

一、前言 在开始学习K8s之前&#xff0c;让我们对容器有一个基本的了解 1.1 什么是容器 一个容器镜像是一个可运行的软件包&#xff0c;其中包含了一个完整的可执行程序&#xff0c;包括代码和运行时需要应用、系统库和全部重要设置的默认值。 通过将应用程序本身&#xff…...

xshell6运行报错:由于找不到mfc110u.dll、MSVCR110.dll无法继续执行代码

今天给大家分享一下我刚装完系统遇到得问题,由于新盟的罗建雨【胡巴】老师帮我给电脑加了固态,又重装了系统,因此电脑里面得所有软件需要重装,在我重装的过程中遇到了一个小问题给大家分享一下,如果大家以后遇到也方便解决。 问题: 安装Xshell时电脑系统报错:“由于找…...

Baklib知识库管理平台,协助组织提升知识管理水平

随着信息时代和知识经济时代的到来&#xff0c;企业内部信息资料繁多冗杂&#xff0c;知识管理逐渐成为各大企业的重要工作之一&#xff0c;企业管理者无不感受到巨大的压力&#xff0c;怎么样将知识进行有效的管理&#xff0c;成为一个难点&#xff0c;并且随着信息不断的更迭…...

一文搞懂core-scheduling核心机制

cookie的原理借助于unsigned long型&#xff0c;和refcount_t引用计数器。 32位64位char *4字节8字节unsigned long4字节8字节 数据结构修改 首先看看实现core scheduling功能对数据结构有哪些修改 task_struct struct task_struct{struct rb_node core_node;unsigned long…...

IP地址在金融行业有哪些应用?

中国加入WTO以来经济得到迅速发展&#xff0c;金融行业随着经济发展体系越来越完善。随着西方金融公司和理念的加入中国金融行业开始多样化发展。金融行业在快速发展的同时也引发了许多弊端。如何维护挖掘客户更大需求&#xff1f;如何获取更多优质客户&#xff1f;如何提升网络…...

GT-suite v2016解决许可证过期问题(附新版liscense下载地址)

安装GT-suite v2016时遇到了如图报错的问题。当时的报错找不到了&#xff0c;下图是贴吧相同问题的报错图。 为了解决问题&#xff0c;先根据某网友的如下答复操作&#xff1a; 添加环境变量后仍然有相同报错。 看来需要寻找其他方法。 再尝试着卸载GT-suite v2016&#xff0c…...

小红书商业笔记与普通笔记区别是什么?小红书笔记有哪几种

主攻单一平台&#xff0c;如何迅速打造爆文。针对软文发布类别的选择&#xff0c;小红书商业笔记与普通笔记区别究竟是什么&#xff0c;今天为大家带来的详细分析&#xff0c;告诉你该如何用最少的成本&#xff0c;做出“爆文”。1、小红书的笔记类型我们都知道&#xff0c;小红…...

DataWhale-统计学习方法打卡Task01

学习教材《统计学习方法&#xff08;第二版&#xff09;》李航 统计学习方法&#xff08;第2版&#xff09; by...李航 (z-lib.org).pdf https://www.aliyundrive.com/s/maJZ6M9hrTe 点击链接保存&#xff0c;或者复制本段内容&#xff0c;打开「阿里云盘」APP &#xff0c;无…...

Java面试——Spring 事务

目录 1.什么是Spring 事务 2.Spring 事务的开启方式 3.Spring事务的实现方式/原理 4.事务传播机制 5.事务隔离级别 6.事务失效的原因 1.什么是Spring 事务 事务在逻辑上是一组操作&#xff0c;要么执行&#xff0c;要不都不执行。 如下&#xff1a; Begin; insert into…...

Python语言零基础入门教程(十九)

Python 异常处理 python提供了两个非常重要的功能来处理python程序在运行中出现的异常和错误。你可以使用该功能来调试python程序。 1、异常处理 2、断言(Assertions) python标准异常 什么是异常&#xff1f; 异常即是一个事件&#xff0c;该事件会在程序执行过程中发生&…...

重生之我是赏金猎人-SRC漏洞挖掘(一)-某SRC测试系统无脑Getshell

0x01 前言 https://github.com/J0o1ey/BountyHunterInChina 欢迎大佬们点个star 0x02 资产收集到脆弱系统 在某src挖掘过程中&#xff0c;本人通过ssl证书对域名资产进行了收集&#xff0c;通过计算域名对应ip段的权重 整理出其C段资产&#xff0c;进行了批量目录扫描 查看…...

Sciter 结合 PReact 实现组件公共逻辑抽离

Sciter 结合 PReact 实现组件公共逻辑抽离 下面例子是获取鼠标移动位置,将这部分逻辑进行抽离 一、使用高阶组件抽离公共逻辑 import {Component } from ./preact.js; export const HOCFactory = (Component) => {class HOC...

OpenTracing协议规范链接

一、官网链接 OpenTracing specificationhttps://opentracing.io/specification/不过目前OpenTracing项目已归档&#xff0c;不再维护。需要参考OpenTelemetry官网链接 Migrating from OpenTracing | OpenTelemetryBackward compatibility with OpenTracing has been a prior…...

金三银四面试必看,自动化测试如何解决日志问题

前言 前几天在员群里&#xff0c;有同学问了一个自动化测试实践中遇到的问题&#xff1a; 持续集成的自动化用例很多&#xff0c;测试环境日志level为debug&#xff0c;日志量大概40G/每天&#xff0c;定位问题时日志查询很慢&#xff0c;该怎么解决&#xff1f; 这个问题可…...

微信怎么开小店?【企业商家微信开店】

企业商家入局微信做营销已经是经营规划中必须做的一件事了&#xff0c;对于企业商家来说&#xff0c;最简单直接的方式就是开一个微信小店&#xff0c;然后通过自己宣传推广来在微信小店中成商品。那么企业商家在微信怎么开小店呢&#xff1f;下面内容分享给想在微信开店的企业…...

Java 中FastJson的使用【吃透FastJson】

如果不了解JSON格式&#xff0c;建议先看下&#xff1a;JSON数据格式【学习记录】 JSON序列化、反序列化JavaBean的框架有很多&#xff0c;最常见的Jackson、阿里巴巴开源的FastJson、谷歌的GSON、apache提供的json-lib等&#xff0c;下面我们主要来熟悉一下&#xff1a;Java语…...

Redis5.0集群搭建

Redis集群教程 此文重在介绍 Redis5.0 三主三从集群安装&#xff0c;无复杂难懂的概念&#xff0c;若想深入了解集群原理请参考Redis集群规范。 Redis集群介绍 Redis Cluster 提供一种 Redis 安装方式&#xff1a;数据自动在多个 Redis 节点间分片。 Redis Cluster 提供一定…...

继企业级信息系统开发学习1.1 —— Spring配置文件管理Bean

骑士救美计划采用构造方法注入属性值1、创建救美任务类2、创建救美骑士类2、创建救美骑士类3、创建旧救美骑士测试类3、配置救美骑士Bean5、创建新救美骑士测试类采用构造方法注入属性值 1、创建救美任务类 在net.huawei.spring.day01包里创建RescueDamselQuest类 Rescue Da…...

Web 容器、HTTP 服务器 、Servlet 容器区别与联系

首先浏览器发起 HTTP 请求&#xff0c;像早期的时候只会请求一些静态资源&#xff0c;这时候需要一个服务器来处理 HTTP 请求&#xff0c;并且将相应的静态资源返回。 这个服务器叫 HTTP 服务器。 简单点说就是解析请求&#xff0c;然后得知需要服务器上面哪个文件夹下哪个名字…...

[2025CVPR]DeepVideo-R1:基于难度感知回归GRPO的视频强化微调框架详解

突破视频大语言模型推理瓶颈,在多个视频基准上实现SOTA性能 一、核心问题与创新亮点 1.1 GRPO在视频任务中的两大挑战 ​安全措施依赖问题​ GRPO使用min和clip函数限制策略更新幅度,导致: 梯度抑制:当新旧策略差异过大时梯度消失收敛困难:策略无法充分优化# 传统GRPO的梯…...

PPT|230页| 制造集团企业供应链端到端的数字化解决方案:从需求到结算的全链路业务闭环构建

制造业采购供应链管理是企业运营的核心环节&#xff0c;供应链协同管理在供应链上下游企业之间建立紧密的合作关系&#xff0c;通过信息共享、资源整合、业务协同等方式&#xff0c;实现供应链的全面管理和优化&#xff0c;提高供应链的效率和透明度&#xff0c;降低供应链的成…...

CentOS下的分布式内存计算Spark环境部署

一、Spark 核心架构与应用场景 1.1 分布式计算引擎的核心优势 Spark 是基于内存的分布式计算框架&#xff0c;相比 MapReduce 具有以下核心优势&#xff1a; 内存计算&#xff1a;数据可常驻内存&#xff0c;迭代计算性能提升 10-100 倍&#xff08;文档段落&#xff1a;3-79…...

渲染学进阶内容——模型

最近在写模组的时候发现渲染器里面离不开模型的定义,在渲染的第二篇文章中简单的讲解了一下关于模型部分的内容,其实不管是方块还是方块实体,都离不开模型的内容 🧱 一、CubeListBuilder 功能解析 CubeListBuilder 是 Minecraft Java 版模型系统的核心构建器,用于动态创…...

听写流程自动化实践,轻量级教育辅助

随着智能教育工具的发展&#xff0c;越来越多的传统学习方式正在被数字化、自动化所优化。听写作为语文、英语等学科中重要的基础训练形式&#xff0c;也迎来了更高效的解决方案。 这是一款轻量但功能强大的听写辅助工具。它是基于本地词库与可选在线语音引擎构建&#xff0c;…...

日常一水C

多态 言简意赅&#xff1a;就是一个对象面对同一事件时做出的不同反应 而之前的继承中说过&#xff0c;当子类和父类的函数名相同时&#xff0c;会隐藏父类的同名函数转而调用子类的同名函数&#xff0c;如果要调用父类的同名函数&#xff0c;那么就需要对父类进行引用&#…...

实战三:开发网页端界面完成黑白视频转为彩色视频

​一、需求描述 设计一个简单的视频上色应用&#xff0c;用户可以通过网页界面上传黑白视频&#xff0c;系统会自动将其转换为彩色视频。整个过程对用户来说非常简单直观&#xff0c;不需要了解技术细节。 效果图 ​二、实现思路 总体思路&#xff1a; 用户通过Gradio界面上…...

6️⃣Go 语言中的哈希、加密与序列化:通往区块链世界的钥匙

Go 语言中的哈希、加密与序列化:通往区块链世界的钥匙 一、前言:离区块链还有多远? 区块链听起来可能遥不可及,似乎是只有密码学专家和资深工程师才能涉足的领域。但事实上,构建一个区块链的核心并不复杂,尤其当你已经掌握了一门系统编程语言,比如 Go。 要真正理解区…...

Axure零基础跟我学:展开与收回

亲爱的小伙伴,如有帮助请订阅专栏!跟着老师每课一练,系统学习Axure交互设计课程! Axure产品经理精品视频课https://edu.csdn.net/course/detail/40420 课程主题:Axure菜单展开与收回 课程视频:...

生产管理系统开发:专业软件开发公司的实践与思考

生产管理系统开发的关键点 在当前制造业智能化升级的转型背景下&#xff0c;生产管理系统开发正逐步成为企业优化生产流程的重要技术手段。不同行业、不同规模的企业在推进生产管理数字化转型过程中&#xff0c;面临的挑战存在显著差异。本文结合具体实践案例&#xff0c;分析…...