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

23种设计模式-责任链模式(Android开发实际应用场景介绍)

什么是责任链模式

责任链模式是一种行为型设计模式,它的核心思想是将请求从一系列处理者中传递,直到其中一个处理者能够处理它为止。在这个过程中,请求可以被任何一个处理者处理,也可以被拒绝,直到有一个处理者能够处理请求。

在Java开发中,责任链模式的实现方式通常是通过一个抽象处理者(Handler)类来定义处理者的接口,并在这个类中定义一个指向下一个处理者的引用,形成一个链式结构。具体处理者(Concrete Handler)则继承自抽象处理者,并实现其处理请求的方法。当一个请求被发起时,它会首先被传递到第一个处理者,如果第一个处理者不能处理它,那么请求会被传递到下一个处理者,直到有一个处理者能够处理它为止。

责任链模式代码解释

下面我们以一个简单的例子来说明责任链模式的使用方法。假设我们有一个请假系统,需要根据不同的请假天数来分别由部门经理、人事主管和总经理进行审核。如果请假天数不超过2天,由部门经理审核;如果请假天数不超过5天,由人事主管审核;如果请假天数超过5天,则需要总经理审核。

首先我们定义一个抽象处理者类Handler:

public abstract class Handler {protected Handler successor;public void setSuccessor(Handler successor) {this.successor = successor;}public abstract void handleRequest(int days);
}

在这个类中,我们定义了一个指向下一个处理者的引用successor,并提供了一个设置下一个处理者的方法setSuccessor。同时,我们还定义了一个抽象方法handleRequest,该方法用于处理请求。具体的处理者将继承这个类,并实现自己的handleRequest方法。

下面我们来定义具体的处理者类。首先是部门经理处理者:

public class DepartmentManager extends Handler {@Overridepublic void handleRequest(int days) {if (days <= 2) {System.out.println("部门经理审批通过");} else {if (successor != null) {successor.handleRequest(days);}}}
}

在这个类中,我们实现了父类中定义的handleRequest方法,如果请假天数不超过2天,则部门经理能够处理该请求并输出相应的处理结果;否则,请求将被传递给下一个处理者,即人事主管处理者。

接下来是人事主管处理者:

public class HRManager extends Handler {@Overridepublic void handleRequest(int days) {if (days <= 5) {System.out.println("人事主管审批通过");} else {if (successor != null) {successor.handleRequest(days);}}}
}

这个类的实现与部门经理处理者类类似,如果请假天数不超过5天,则人事主管能够处理该请求并输出相应的处理结果;否则,请求将被传递给下一个处理者,即总经理处理者。

最后是总经理处理者:

public class GeneralManager extends Handler {@Overridepublic void handleRequest(int days) {System.out.println("总经理审批通过");}
}

这个类的实现非常简单,因为总经理是责任链中的最后一个处理者,无论请求的请假天数是多少,总经理都能够处理该请求。

最后我们来测试一下这个责任链模式的实现:

public class Test {public static void main(String[] args) {Handler departmentManager = new DepartmentManager();Handler hrManager = new HRManager();Handler generalManager = new GeneralManager();departmentManager.setSuccessor(hrManager);hrManager.setSuccessor(generalManager);departmentManager.handleRequest(1);departmentManager.handleRequest(3);departmentManager.handleRequest(7);}
}

在这个测试程序中,我们首先创建了部门经理、人事主管和总经理三个处理者,然后通过setSuccessor方法将它们组合成一个责任链。最后我们分别测试了请求请假1天、请假3天和请假7天的情况。运行结果如下:

部门经理审批通过
人事主管审批通过
总经理审批通过

可以看到,在请求请假1天、请假3天和请假7天的情况下,责任链分别被部门经理、人事主管和总经理处理,并输出了相应的处理结果。

Android开发中的实际应用解析

在安卓中,责任链模式也有广泛的应用。例如,Android中的事件分发机制就是一种典型的责任链模式。当用户触摸屏幕时,事件首先会被传递给Activity,在Activity中处理完后,事件会继续被传递给它的子View,由子View来处理。如果子View不能处理该事件,事件会被传递给父View,直到事件被处理为止。这个过程形成了一个责任链,而每个View都是责任链中的一个处理者。下面我们来看一下Android源码中的事件分发机制实现。

Android中的事件分发机制是从ViewGroup类开始的。ViewGroup类继承自View类,是一种特殊的View,可以包含其他View。在ViewGroup类中,有一个onInterceptTouchEvent方法,用于拦截子View的触摸事件。如果一个ViewGroup在onInterceptTouchEvent方法中返回了true,则表示它要拦截子View的触摸事件,并将事件交给自己的onTouchEvent方法来处理。如果返回了false,则表示它不拦截子View的触摸事件,将事件传递给子View来处理。

下面是ViewGroup类中onInterceptTouchEvent方法的源码:

public boolean onInterceptTouchEvent(MotionEvent ev) {if (ev.isFromSource(InputDevice.SOURCE_MOUSE)&& ev.getActionMasked() == MotionEvent.ACTION_DOWN&& ev.isButtonPressed(MotionEvent.BUTTON_SECONDARY)) {// The right mouse button is currently pressed.// We want to give the user a chance to reconsider before// performing an action that cannot be undone, such as dismissing// a notification or deleting a message.showContextMenu();return true;}if (mOnInterceptTouchListener != null && mOnInterceptTouchListener.onTouch(this, ev)) {return true;}if (ev.isFromSource(InputDevice.SOURCE_CLASS_POINTER)) {final int action = ev.getAction();final float x = ev.getX();final float y = ev.getY();if (action == MotionEvent.ACTION_DOWN) {mLastTouchDownIndex = ev.getActionIndex();mLastTouchDownTime = ev.getDownTime();mLastTouchDownX = x;mLastTouchDownY = y;mIgnoreNextUpEvent = false;} else {// Short circuit if we've already canceled, or if we have no// children.final boolean canceled = (mGroupFlags & FLAG_DISALLOW_INTERCEPT) != 0;if (!canceled && mChildrenCount != 0) {// If this is a pointer event, propagate it to our children.if (action == MotionEvent.ACTION_POINTER_DOWN|| action == MotionEvent.ACTION_MOVE) {final int index = ev.getActionIndex();final int id = ev.getPointerId(index);if (id != mScrollPointerId) {final float dx = x - mLastTouchDownX;final float dy = y - mLastTouchDownY;final float xDiff = Math.abs(dx);final float yDiff = Math.abs(dy);final int touchSlop = mTouchSlop;final boolean isGeneratedGesture = ev.isGeneratedGesture();boolean interceptHorizontal = false;boolean interceptVertical = false;if ((mGroupFlags & FLAG_DISALLOW_INTERCEPT) == 0) {if ((mNestedScrollAxes & SCROLL_AXIS_HORIZONTAL) != 0) {interceptHorizontal |= (xDiff > touchSlop);}if ((mNestedScrollAxes & SCROLL_AXIS_VERTICAL) != 0) {interceptVertical |= (yDiff > touchSlop);}}if (isGeneratedGesture|| interceptHorizontal|| interceptVertical|| hasPointersOutOfView()) {mLastTouchMotionY = y;mLastTouchMotionX = x;cancelAndClearTouchTargets(ev);return true;}}} else if (action == MotionEvent.ACTION_POINTER_UP) {final int index = ev.getActionIndex();final int id = ev.getPointerId(index);if (id == mScrollPointerId) {// This was our active pointer going up. Choose a new// action pointer and adjust accordingly.final int newIndex = ev.getPointerCount() - 1;mLastTouchDownX = ev.getX(newIndex);mLastTouchDownY = ev.getY(newIndex);mLastTouchDownIndex = ev.findPointerIndex(newIndex);if (mVelocityTracker != null) {mVelocityTracker.clear();}}}}}}return false;
}

可以看到,在onInterceptTouchEvent方法中,首先判断了是否有设置OnInterceptTouchListener,如果有则交由OnInterceptTouchListener来处理触摸事件。

接着,如果触摸事件是鼠标右键的按下事件,则弹出上下文菜单并返回true,拦截触摸事件,防止后续的触摸事件继续传递下去。

如果以上两种情况都不满足,则判断触摸事件是否来自于指针,如果是,则根据具体的触摸事件类型进行处理。例如,如果是按下事件,则记录下按下的坐标和时间等信息;如果是移动事件,则计算移动距离,并根据一定的条件来决定是否拦截事件。

如果没有任何一个子View对该事件进行了处理,则返回false,表示不拦截该事件,让其继续传递给子View进行处理。

以上就是在安卓中使用责任链模式的一个例子,即事件分发机制。在这个机制中,每个子View都是一个责任链节点,它们依次处理触摸事件,并根据具体的情况来决定是否将事件交给下一个节点处理。

总结

责任链模式的优点在于,它能够将一个大型的、复杂的任务分解成多个小的、简单的任务,每个任务都由一个节点来处理,从而降低了系统的复杂性和耦合度。此外,责任链模式还具有灵活性,可以根据具体的需求来动态地调整节点的顺序和个数,从而满足不同的业务场景。

相关文章:

23种设计模式-责任链模式(Android开发实际应用场景介绍)

什么是责任链模式 责任链模式是一种行为型设计模式&#xff0c;它的核心思想是将请求从一系列处理者中传递&#xff0c;直到其中一个处理者能够处理它为止。在这个过程中&#xff0c;请求可以被任何一个处理者处理&#xff0c;也可以被拒绝&#xff0c;直到有一个处理者能够处…...

Socket+Select+Epoll笔记

讲到epoll&#xff0c;就必须了解Socket&#xff0c;上篇博客写了Socket的基本使用方法&#xff0c;步骤主要为创建一个socketsocket是进程之间通信的&#xff0c;那么进程通信如何找到这个socket呢&#xff1f;当然是端口号&#xff0c;所以socket就要和端口号进行绑定&#x…...

git查看最近修改的文件

git log --name-status 每次修改的文件列表, 显示状态 git log --name-only 每次修改的文件列表 git log --stat 每次修改的文件列表, 及文件修改的统计 git whatchanged 每次修改的文件列表 git whatchanged --stat 每次修改的文件列表, 及文件修改的统计 git show 显示最…...

【算法基础(四)】堆排序(二)

堆排序&#xff08;二&#xff09; 把数组从零开始连续的一段 完全二叉树 size i 左 son 2*11 i 右 son 2*12 父 (i-1) / 2 堆是完全二叉树&#xff0c;分为大根堆和小根堆 在完全二叉树里&#xff0c;每一棵子数最大的值是头节点的值&#xff0c;就是大根堆 同理&…...

C++类型转换

C语言的转换是在变量前加类型名进行转换的&#xff0c;比如double pi 3.14;int a (int) pi;对于指针也是如此double* dptr &pi;int* iptr (int*)dptr;虽然c兼容了C语言的转型方式&#xff0c;但是也做了很多限制&#xff0c;比如向上类型转换&#xff0c;在c中建议使用…...

Keil MDK6要来了,将嵌入式软件开发水平带到新高度,支持跨平台(2023-03-11)

注&#xff1a;这个是MDK6&#xff0c;不是MDK5 AC6&#xff0c;属于下一代MDK视频版&#xff1a; https://www.bilibili.com/video/BV16s4y157WF Keil MDK6要来了&#xff0c;将嵌入式软件开发水平带到新高度&#xff0c;支持跨平台一年一度的全球顶级嵌入式会展Embedded Wor…...

蓝桥杯刷题第九天

题目描述本题为填空题&#xff0c;只需要算出结果后&#xff0c;在代码中使用输出语句将所填结果输出即可。素数就是不能再进行等分的整数。比如7&#xff0c;11。而 9 不是素数&#xff0c;因为它可以平分为 3 等份。一般认为最小的素数是2&#xff0c;接着是 3&#xff0c;5&…...

a-tree-select 基本使用,下拉框高度和宽度设置、回显时滚动条定位解决。

目录一、基本使用1. 界面效果2. 代码实现3. 问题1&#xff1a;下拉框占满整个屏幕4. 问题4&#xff1a;菜单内容过长时&#xff0c;下拉菜单宽度无限变宽。二、数据回显、滚动条定位1. 界面效果2. 代码实现2.1 获取默认展开节点2.1.1 代码实现2.1.2 说明2.2 设置滚动条定位2.2.…...

【Linux】之nc命令(连接与扫描指定端口、监测服务端口的使用情况)解析、详解实例、邮件告警

&#x1f341;博主简介 &#x1f3c5;云计算领域优质创作者   &#x1f3c5;华为云开发者社区专家博主   &#x1f3c5;阿里云开发者社区专家博主 &#x1f48a;交流社区&#xff1a;运维交流社区 欢迎大家的加入&#xff01; 文章目录nc命令简介nc命令的安装nc命令语法格式…...

cdn简单配置

cdn配置域名接入CDN编辑CDN配置本地修改hosts文件&#xff0c;绕过公网解析域名接入CDN 添加CDN域名以及回源配置 编辑CDN配置 默认后端端口是80&#xff0c;如果测试发现无法访问&#xff0c;则可能是443或其它 如果域名在CDN后端有https强制跳转&#xff0c;后端端口一定是44…...

前端安全(自留)

目录XSS——跨站脚本常见解决CSRF ——跨站请求伪造常见解决XSS——跨站脚本 当目标站点在渲染html的过程中&#xff0c;遇到陌生的脚本指令执行。 攻击者通过在网站注入恶意脚本&#xff0c;使之在用户的浏览器上运行&#xff0c;从而盗取用户的信息如 cookie 等。 常见 解…...

零基础转行云计算可行吗

目前处于云年代&#xff0c;云计算运维工程师的工作远景还是十分广泛的。像是阿里云计算&#xff0c;滴滴&#xff0c;抖音等等互联网大厂目前都在使用云核算技能。 云计算运维工程师的薪资水平也十分可观。 运维工程师(Operations)&#xff0c;在国内又称为运维开发工程师(Dev…...

【AcWing】蓝桥杯备赛-深度优先搜索-dfs(1)

目录 写在前面&#xff1a; 题目&#xff1a;92. 递归实现指数型枚举 - AcWing题库 读题&#xff1a; 输入格式&#xff1a; 输出格式&#xff1a; 数据范围&#xff1a; 输入样例&#xff1a; 输出样例&#xff1a; 解题思路&#xff1a; 代码&#xff1a; AC &…...

孩子免费就读|私企经理自费赴美国东海岸高校访学

私企U经理无文章无课题&#xff0c;出国访学除了为考察市场、拓宽人脉、提升履资外&#xff0c;另一个主要目的是带孩子在美国接受当地免费的公立中小学教育&#xff0c;并把访学目标学校定位在东海岸。最终其采纳了板凳费相对较低的佐治亚大学邀请函&#xff0c;签证时居然全家…...

前端面试hr经常会问的问题

文章目录前言1.自我介绍2.为什么你要离职&#xff1f;3.工作经历4.职业规划5.优点、缺点6.还有什么要问的总结前言 这里记录了一些面试中hr或者项目负责人经常会问的一些问题&#xff0c;可以提前参考参考&#xff0c;想想该怎么回答&#xff0c;为之后的面试做好准备&#xf…...

C动态数组

在实际项目中&#xff0c;我们经常与各式各样的数据打交道。 例如&#xff1a;我们处理的是学生的数据。 struct student {int id; // 学号char name[20]; // 姓名int gender; // 性别int mark; // 成绩 };学生数据使用一个结构体表示&#xff0c;该结构体拥有4个成员。分别为…...

【STL一】STL组件(容器、迭代器、算法)

【STL一】STL组件&#xff08;容器、迭代器、算法&#xff09;一、STL二、STL组件&#xff08;component&#xff09;1、stl六大组件2、C STL的13个头文件3、stl所有头文件三、容器&#xff08;container)1、序列容器&#xff08;Sequence container)——顺序容器2、关联容器&a…...

Java每日一练(20230312)

目录 1. 两数之和 II ★ 2. 反转链表 ★★ 3. 二叉树的层序遍历 II ★★★ &#x1f31f; 每日一练刷题专栏 C/C 每日一练 ​专栏 Python 每日一练 专栏 Java 每日一练 专栏 1. 两数之和 II 给定一个已按照 非递减顺序排列 的整数数组 numbers &#xff0c;请你从数…...

Linux中sudo,su与su -命令的区别

前言 su命令就是切换用户的工具&#xff0c;怎么理解呢&#xff1f;比如我们以普通用户tom登录的&#xff0c;但要添加用户任务&#xff0c;执行useradd &#xff0c;tom用户没有这个权限&#xff0c;而这个权限恰恰由root所拥有。解决办法无法有两个&#xff0c;一是退出tom用…...

归并排序有多简单?一幅图教你看懂【C语言】

目录 归并排序的递归实现 代码实现 归并排序的非递归实现 代码实现 归并排序的思想很简单——分治法。简单地说&#xff0c;归并排序的是将序列拆分成几段子序列&#xff0c;将每一段子序列分别进行排序&#xff0c;排好之后再将有序的子序列归并&#xff08;有点像合并两…...

极简安装方案:树莓派部署OpenClaw轻量版对接云端Qwen3-32B

极简安装方案&#xff1a;树莓派部署OpenClaw轻量版对接云端Qwen3-32B 1. 为什么选择树莓派OpenClaw轻量版&#xff1f; 去年夏天&#xff0c;我突发奇想&#xff1a;能不能用树莓派做个24小时在线的AI管家&#xff1f;既能控制智能家居&#xff0c;又能处理简单办公任务。但…...

快速验证控制逻辑:用快马平台十分钟搭建pid算法仿真原型

今天想和大家分享一个快速验证PID控制算法的小技巧。作为一名自动化工程师&#xff0c;经常需要调试各种控制参数&#xff0c;传统方法要搭建物理实验环境或者用MATLAB仿真&#xff0c;都很费时。最近发现用InsCode(快马)平台可以十分钟就做出一个可交互的PID仿真原型&#xff…...

Python异步I/O终极调优手册(含strace+py-spy+asyncio debug mode三重追踪链路图)

第一章&#xff1a;Python异步I/O性能瓶颈的本质洞察Python的async/await语法虽大幅简化了异步编程模型&#xff0c;但其底层性能瓶颈并非源于语法糖本身&#xff0c;而根植于事件循环调度机制、GIL对CPU密集型任务的制约&#xff0c;以及I/O等待与协程切换之间的隐式开销。事件…...

Windows Server远程管理新选择:一键脚本部署noVNC服务端(含开机自启配置)

Windows Server远程管理新选择&#xff1a;一键脚本部署noVNC服务端&#xff08;含开机自启配置&#xff09; 对于需要管理Windows Server的系统管理员来说&#xff0c;远程访问是不可或缺的功能。传统的RDP虽然稳定&#xff0c;但在某些场景下可能受限&#xff0c;比如网络环境…...

AI的“血管”:从大模型需求看6G、高速光纤与智算中心网络的技术变革

大模型训练与推理的爆发&#xff0c;正以前所未有的力度重塑通信网络基础设施。6G、高速光纤、智算中心网络&#xff0c;正成为AI基础设施的“血管”&#xff0c;承载着算力的血液&#xff0c;决定智能的极限。当GPT-5.4的推理能力逼近人类专家&#xff0c;当Sora可以生成一分钟…...

3000 字深度拆解:Paperxie AI 期刊写作界面全解析 —— 科研人必看的 “投刊效率密码”

paperxie-免费查重复率aigc检测/开题报告/毕业论文/智能排版/文献综述/期刊论文https://www.paperxie.cn/ai/journalArticleshttps://www.paperxie.cn/ai/journalArticles 一、引言&#xff1a;科研人的投稿困局&#xff0c;藏在每一个被忽略的界面细节里 当科研人熬过无数个深…...

微信小程序人脸核身接入全攻略:从资质准备到代码实现(附避坑指南)

微信小程序人脸核身接入实战&#xff1a;合规指南与代码精要 在金融、政务等高安全要求的场景中&#xff0c;确保用户身份真实性已成为刚需。微信小程序提供的人脸核身能力&#xff0c;将活体检测、OCR识别与权威数据比对融为一体&#xff0c;为开发者提供了合规且高效的身份验…...

CREST:如何用5分钟开启分子构象探索之旅?

CREST&#xff1a;如何用5分钟开启分子构象探索之旅&#xff1f; 【免费下载链接】crest Conformer-Rotamer Ensemble Sampling Tool based on the xtb Semiempirical Extended Tight-Binding Program Package 项目地址: https://gitcode.com/gh_mirrors/crest/crest 在…...

大模型RAG入门基础架构介绍

传统大模型的局限性 知识可能过时&#xff08;训练数据有时效 性&#xff09;会产生"幻觉"&#xff08;编造不存在的信息&#xff09;无法访问私有知识库数据回答缺乏具体出处&#xff0c;难以验证最大对话上下文限制&#xff08;大部分模型128K&#xff09; RAG的…...

告别蓝牙!用STM32F103和NRF24L01搭建低成本2.4G无线通信,实测传输距离与稳定性

STM32F103与NRF24L01构建高性能2.4G私有通信系统实战指南 在物联网设备爆发式增长的今天&#xff0c;无线通信模块的选择成为硬件开发者面临的首要难题。面对市面上琳琅满目的蓝牙、Wi-Fi和私有协议模块&#xff0c;如何根据项目需求选择最具性价比的解决方案&#xff1f;本文将…...