责任链模式:优雅处理复杂流程的设计艺术
引言
在软件设计中,我们经常会遇到需要按特定顺序处理请求的场景。例如,一个订单处理系统可能需要经过验证、付款、物流安排和客户通知等多个步骤。如果我们将这些步骤硬编码在一个方法中,代码将变得臃肿且难以维护。这时,责任链模式(Chain of Responsibility Pattern)就显示出了它的价值。
责任链模式是一种行为设计模式,它允许多个对象依次处理同一个请求,从而将请求的发送者与接收者解耦。本文将详细介绍责任链模式的核心概念、实现方法以及适用场景。
什么是责任链模式?
责任链模式是这样工作的:一个请求沿着一条预定义的对象链传递,直到有一个对象处理它或者请求到达链尾。每个处理者决定是自己处理请求还是将其传递给链中的下一个对象。
这种模式的美妙之处在于:请求的发送者不需要知道谁会处理这个请求,而每个处理者也只需要关注自己的职责,无需知道整个链的结构。
责任链模式的核心组件
- 抽象处理者(Handler): 定义了处理请求的接口,通常包含设置下一个处理者和处理请求的方法。
- 具体处理者(ConcreteHandler): 实现抽象处理者的接口,处理自己负责的请求,如果不能处理则传递给下一个处理者。
- 客户端(Client): 创建处理者对象并组成责任链,然后向链上的第一个处理者发送请求。
类图
下面是责任链模式的类图:
责任链模式的实现
让我们以一个订单处理系统为例,说明如何实现责任链模式:
1. 创建订单类
/*** 订单类 - 包含订单的基本信息和处理状态*/
public class Order {private int id; // 订单IDprivate double amount; // 订单金额private String customerName; // 客户名称private boolean isValidated; // 订单是否已验证private boolean isPaymentProcessed; // 支付是否已处理private boolean isDeliveryArranged; // 配送是否已安排private boolean isNotificationSent; // 通知是否已发送/*** 订单构造函数* @param id 订单ID* @param amount 订单金额* @param customerName 客户名称*/public Order(int id, double amount, String customerName) {this.id = id;this.amount = amount;this.customerName = customerName;this.isValidated = false;this.isPaymentProcessed = false;this.isDeliveryArranged = false;this.isNotificationSent = false;}// Getters and setterspublic int getId() {return id;}public double getAmount() {return amount;}public String getCustomerName() {return customerName;}public boolean isValidated() {return isValidated;}public void setValidated(boolean validated) {isValidated = validated;}public boolean isPaymentProcessed() {return isPaymentProcessed;}public void setPaymentProcessed(boolean paymentProcessed) {this.isPaymentProcessed = paymentProcessed;}public boolean isDeliveryArranged() {return isDeliveryArranged;}public void setDeliveryArranged(boolean deliveryArranged) {this.isDeliveryArranged = deliveryArranged;}public boolean isNotificationSent() {return isNotificationSent;}public void setNotificationSent(boolean notificationSent) {this.isNotificationSent = notificationSent;}@Overridepublic String toString() {return "Order{" +"id=" + id +", amount=" + amount +", customerName='" + customerName + '\'' +", isValidated=" + isValidated +", isPaymentProcessed=" + isPaymentProcessed +", isDeliveryArranged=" + isDeliveryArranged +", isNotificationSent=" + isNotificationSent +'}';}
}
2. 创建处理器接口
/*** 订单处理器抽象类 - 责任链模式的核心* 定义了处理订单的通用接口和设置下一个处理器的方法*/
public abstract class OrderHandler {protected OrderHandler nextHandler; // 责任链中的下一个处理器/*** 设置责任链中的下一个处理器* @param nextHandler 下一个处理器*/public void setNextHandler(OrderHandler nextHandler) {this.nextHandler = nextHandler;}/*** 处理订单的抽象方法,由具体的处理器实现* @param order 要处理的订单*/public abstract void handleOrder(Order order);
}
3. 实现具体的处理器类
订单验证处理器
/*** 订单验证处理器 - 责任链的第一环* 负责验证订单的基本信息是否有效*/
public class OrderValidationHandler extends OrderHandler {@Overridepublic void handleOrder(Order order) {System.out.println("OrderValidationHandler: 验证订单 #" + order.getId());// 验证订单逻辑if (order.getAmount() <= 0) {System.out.println("OrderValidationHandler: 订单金额无效,处理终止");return; // 终止责任链,不再继续处理}if (order.getCustomerName() == null || order.getCustomerName().isEmpty()) {System.out.println("OrderValidationHandler: 客户信息无效,处理终止");return; // 终止责任链,不再继续处理}// 标记为已验证order.setValidated(true);System.out.println("OrderValidationHandler: 订单 #" + order.getId() + " 已验证通过");// 传递给下一个处理器if (nextHandler != null) {nextHandler.handleOrder(order); // 继续责任链的处理流程}}
}
支付处理器
/*** 支付处理器 - 责任链的第二环* 负责处理订单的支付流程*/
public class PaymentProcessHandler extends OrderHandler {@Overridepublic void handleOrder(Order order) {// 检查订单是否已验证,确保责任链的顺序性if (!order.isValidated()) {System.out.println("PaymentProcessHandler: 订单 #" + order.getId() + " 未通过验证,无法处理支付");return; // 前置条件不满足,终止处理}System.out.println("PaymentProcessHandler: 处理订单 #" + order.getId() + " 的支付,金额: " + order.getAmount());// 模拟支付处理逻辑// 这里可以加入支付网关调用等实际业务逻辑boolean paymentSuccessful = processPayment(order);if (paymentSuccessful) {order.setPaymentProcessed(true);System.out.println("PaymentProcessHandler: 订单 #" + order.getId() + " 支付成功");// 传递给下一个处理器if (nextHandler != null) {nextHandler.handleOrder(order); // 继续责任链的处理流程}} else {System.out.println("PaymentProcessHandler: 订单 #" + order.getId() + " 支付失败,处理终止");// 支付失败,终止责任链}}/*** 处理支付的内部方法* @param order 要处理支付的订单* @return 支付是否成功*/private boolean processPayment(Order order) {// 模拟支付处理// 在实际应用中,这里会调用支付网关APIreturn order.getAmount() <= 10000; // 假设10000以上的订单需要额外审核}
}
配送安排处理器
/*** 配送安排处理器 - 责任链的第三环* 负责为已支付的订单安排配送*/
public class DeliveryArrangementHandler extends OrderHandler {@Overridepublic void handleOrder(Order order) {// 检查订单是否已支付,确保责任链的顺序性if (!order.isPaymentProcessed()) {System.out.println("DeliveryArrangementHandler: 订单 #" + order.getId() + " 未完成支付,无法安排配送");return; // 前置条件不满足,终止处理}System.out.println("DeliveryArrangementHandler: 为订单 #" + order.getId() + " 安排配送");// 模拟配送安排逻辑arrangeDelivery(order);order.setDeliveryArranged(true);System.out.println("DeliveryArrangementHandler: 订单 #" + order.getId() + " 已安排配送");// 传递给下一个处理器if (nextHandler != null) {nextHandler.handleOrder(order); // 继续责任链的处理流程}}/*** 安排配送的内部方法* @param order 要安排配送的订单*/private void arrangeDelivery(Order order) {// 模拟配送安排// 在实际应用中,这里会与物流系统对接System.out.println("DeliveryArrangementHandler: 正在为客户 " + order.getCustomerName() + " 安排递送服务");}
}
通知处理器
/*** 通知处理器 - 责任链的最后一环* 负责在订单处理完成后发送通知给客户*/
public class NotificationHandler extends OrderHandler {@Overridepublic void handleOrder(Order order) {// 检查订单是否已安排配送,确保责任链的顺序性if (!order.isDeliveryArranged()) {System.out.println("NotificationHandler: 订单 #" + order.getId() + " 未安排配送,无法发送通知");return; // 前置条件不满足,终止处理}System.out.println("NotificationHandler: 发送订单 #" + order.getId() + " 的通知");// 模拟发送通知逻辑sendNotification(order);order.setNotificationSent(true);System.out.println("NotificationHandler: 订单 #" + order.getId() + " 的通知已发送");// 完成处理,这是责任链的最后一环System.out.println("订单 #" + order.getId() + " 处理完成!");}/*** 发送通知的内部方法* @param order 要发送通知的订单*/private void sendNotification(Order order) {// 模拟发送通知// 在实际应用中,这里会调用邮件或短信服务System.out.println("NotificationHandler: 向客户 " + order.getCustomerName() + " 发送订单确认通知");}
}
4. 创建客户端测试类
/*** 订单处理器 - 客户端类* 负责构建和初始化责任链,并启动订单处理流程*/
public class OrderProcessor {private OrderHandler chain; // 责任链的起点/*** 构造函数,初始化责任链*/public OrderProcessor() {// 构建责任链buildOrderProcessingChain();}/*** 构建订单处理责任链* 设定处理器的顺序:验证 -> 支付 -> 配送 -> 通知*/private void buildOrderProcessingChain() {// 创建各个处理器OrderHandler validationHandler = new OrderValidationHandler();OrderHandler paymentHandler = new PaymentProcessHandler();OrderHandler deliveryHandler = new DeliveryArrangementHandler();OrderHandler notificationHandler = new NotificationHandler();// 设置处理器链,定义处理顺序validationHandler.setNextHandler(paymentHandler);paymentHandler.setNextHandler(deliveryHandler);deliveryHandler.setNextHandler(notificationHandler);// 设置责任链的起点this.chain = validationHandler;}/*** 处理订单的方法,将订单传入责任链的起点* @param order 要处理的订单*/public void processOrder(Order order) {System.out.println("开始处理订单 #" + order.getId());chain.handleOrder(order); // 启动责任链处理}
}
5. 主类用于运行示例
/*** 责任链模式演示类* 用于测试责任链在不同订单场景下的行为*/
public class ChainOfResponsibilityDemo {public static void main(String[] args) {// 创建订单处理器OrderProcessor processor = new OrderProcessor();// 测试案例1: 有效订单 - 应该完成所有处理步骤Order validOrder = new Order(1001, 1200.50, "张三");System.out.println("=== 处理有效订单 ===");processor.processOrder(validOrder);System.out.println("最终订单状态: " + validOrder);System.out.println();// 测试案例2: 金额无效的订单 - 应在验证阶段终止Order invalidAmountOrder = new Order(1002, 0, "李四");System.out.println("=== 处理金额无效的订单 ===");processor.processOrder(invalidAmountOrder);System.out.println("最终订单状态: " + invalidAmountOrder);System.out.println();// 测试案例3: 客户信息为空的订单 - 应在验证阶段终止Order noCustomerOrder = new Order(1003, 2500.75, "");System.out.println("=== 处理客户信息为空的订单 ===");processor.processOrder(noCustomerOrder);System.out.println("最终订单状态: " + noCustomerOrder);System.out.println();}
}
6. 测试结果
责任链模式的优点
- 降低耦合度: 发送者与接收者解耦,发送者不需要知道谁会处理请求。
- 增加灵活性: 可以动态地改变责任链的成员或者调整处理顺序。
- 单一职责: 每个处理者只关注自己的职责,符合单一职责原则。
- 开闭原则: 可以在不修改现有代码的情况下增加新的处理者,符合开闭原则。
责任链模式的缺点
- 性能考量: 请求可能需要经过多个处理者才能得到处理,增加了处理时间。
- 保证处理: 如果责任链设计不当,可能导致请求无法被处理。
- 调试困难: 请求的流向可能不明确,增加了调试难度。
责任链模式的应用场景
责任链模式适用于以下场景:
- 多个对象可以处理同一个请求,但具体由哪个对象处理在运行时确定。
- 需要动态指定一组对象处理请求。
- 不明确指定请求处理者的情况下,向多个对象中的一个提交请求。
实际应用例子:
- Web应用中的过滤器和拦截器
- 日志记录系统的级别处理
- 异常处理机制
- 工作流系统中的审批流程
总结
责任链模式通过将请求沿着处理者链进行传递,实现了请求发送者与接收者之间的解耦。它不仅使代码更加整洁和模块化,还提高了系统的灵活性和可维护性。在处理复杂流程和事件传递时,责任链模式是一个非常有价值的设计工具。
当你的系统中出现一系列处理者,它们以特定顺序处理请求,且请求的处理流程灵活多变时,请考虑使用责任链模式。它将帮助你构建更加优雅和可扩展的系统架构。
相关文章:
责任链模式:优雅处理复杂流程的设计艺术
引言 在软件设计中,我们经常会遇到需要按特定顺序处理请求的场景。例如,一个订单处理系统可能需要经过验证、付款、物流安排和客户通知等多个步骤。如果我们将这些步骤硬编码在一个方法中,代码将变得臃肿且难以维护。这时,责任链…...
【DeepSeek】5分钟快速实现本地化部署教程
一、快捷部署 (1)下载ds大模型安装助手,下载后直接点击快速安装即可。 https://file-cdn-deepseek.fanqiesoft.cn/deepseek/deepseek_28348_st.exe (2)打开软件,点击立即激活 (3)选…...
HTML前端手册
HTML前端手册 记录前端框架在使用过程中遇到的各种问题和解决方案,供后续快速进行手册翻阅使用 文章目录 HTML前端手册1-前端框架1-TypeScript框架2-CSS框架 2-前端Demo1-Html常用代码 2-知云接力3-Live2D平面动画 3-前端运维1-NPM版本管理 1-前端框架 1-TypeScrip…...
【uniapp】图片添加canvas水印
目录 需求&背景实现地理位置添加水印 ios补充 需求&背景 需求:拍照后给图片添加水印, 水印包含经纬度、用户信息、公司logo等信息。 效果图: 方案:使用canvas添加水印。 具体实现:上传图片组件是项目里现有的ÿ…...
Java 大视界 -- Java 大数据在智能金融反欺诈中的技术实现与案例分析(114)
💖亲爱的朋友们,热烈欢迎来到 青云交的博客!能与诸位在此相逢,我倍感荣幸。在这飞速更迭的时代,我们都渴望一方心灵净土,而 我的博客 正是这样温暖的所在。这里为你呈上趣味与实用兼具的知识,也…...
机器学习数学基础:43.外生变量与内生变量
外生变量与内生变量:模型中的因果角色 在因果模型(像结构方程模型、回归分析这类)里,外生变量和内生变量是用来区分变量来源和相互关系的重要概念。下面从定义、实例、差异以及应用场景四个方面来详细介绍: 一、定义…...
Bean 的生命周期主要包括以下阶段:
Bean 的生命周期主要包括以下阶段: 定义 :在配置文件或注解中定义 Bean,包括其类、作用域等信息。 实例化 :Spring 容器根据定义创建 Bean 的实例。 属性赋值 :容器为 Bean 设置配置的属性值。 初始化 :…...
Unity游戏开发中的网格简化与LOD技术(Mesh Simplification LOD)
在Unity游戏开发中,网格简化(Mesh Simplification)和LOD(Level of Detail)技术是优化渲染性能的关键手段,尤其在处理复杂场景和高精度模型时至关重要。以下是一套系统的实现方案与优化策略: 一、…...
3.7[Q]CV
对于一个由cmake构建的项目,什么时候应该执行cmake指令?什么时候执行make指令?即,一个cmake构建的项目,各步骤的意义是什么?当我修改了部分代码后,重启项目该执行什么命令? view,mod…...
发行思考:全球热销榜的频繁变动
几点杂感: 1、单机游戏销量与在线人数的衰退是剧烈的,有明显的周期性,而在线游戏则稳定很多。 如去年的某明星游戏,最高200多万在线,如今在线人数是48名,3万多。 而近期热门的是MH,在线人数8…...
Springboot全局LocalDateTime时间格式化配置
我们对字段的日期格式化时一般会用注解: JsonFormat(pattern "yyyy-MM-dd HH:mm:ss", timezone "GMT8")private Date createDate;但是每个字段都要写也太麻烦了 不是我的全局化作风 在application.yml中配置全局时间格式化只会对Date类型有用: jackson:d…...
Redis主从复制
目录 点单问题 启动多个redis服务器 配置主从结构 查看主从结构信息 断开主从结构 修改主从结构 主从复制的拓扑结构 主从复制的基本流程 全量复制和部分复制 全量复制的流程 部分复制的流程 实时复制的流程 主从复制总结 主从复制是基于分布式系统进行讨论的&am…...
玩转python:掌握Python数据结构之栈Stack
栈(Stack)是计算机科学中一种非常基础且重要的数据结构。它的特点是后进先出(LIFO,Last In First Out),就像我们生活中叠盘子一样,最后放上去的盘子总是最先被拿走。本文将用通俗易懂的语言和丰…...
电脑如何拦截端口号,实现阻断访问?
如果你弟弟喜欢玩游戏,你可以查询该应用占用的端口,结合以下方法即可阻断端口号,让弟弟好好学习,天天向上! 拦截端口可以通过防火墙和路由器进行拦截 ,以下是常用方法: 方法 1:使用…...
DeepSeek 医疗大模型微调实战讨论版(第一部分)
DeepSeek医疗大模型微调实战指南第一部分 DeepSeek 作为一款具有独特优势的大模型,在医疗领域展现出了巨大的应用潜力。它采用了先进的混合专家架构(MoE),能够根据输入数据的特性选择性激活部分专家,避免了不必要的计算,极大地提高了计算效率和模型精度 。这种架构使得 …...
Apache Httpd 多后缀解析
目录 1.原因 2.环境 3.复现 4.防御 1.Apache Httpd 多后缀解析原因 Apache HTTP Server 在处理文件请求时,通常会根据文件的后缀来确定如何处理该文件。例如,.php文件会被交给 PHP 解释器处理,而.html文件则直接作为静态文件返回。 然而…...
2025年03月07日Github流行趋势
项目名称:ai-hedge-fund 项目地址url:https://github.com/virattt/ai-hedge-fund项目语言:Python历史star数:12788今日star数:975项目维护者:virattt, seungwonme, KittatamSaisaard, andorsk, arsaboo项目…...
Jenkins在Windows上的使用(二):自动拉取、打包、部署
(一)Jenkins全局配置 访问部署好的Jenkins服务器网址localhost:8080,完成默认插件的安装后,接下来将使用SSH登录远程主机以实现自动化部署。 1. 配置插件 选择dashboard->Manage Jenkins->plugins 安装下面两个插件 …...
【JavaEE】-- 多线程(初阶)4
文章目录 8.多线程案例8.1 单例模式8.1.1 饿汉模式8.1.2 懒汉模式 8.2 阻塞队列8.2.1 什么是阻塞队列8.2.2 生产者消费者模型8.2.3 标准库中的阻塞队列8.2.4 阻塞队列的应用场景8.2.4.1 消息队列 8.2.5 异步操作8.2.5 自定义实现阻塞队列8.2.6 阻塞队列--生产者消费者模型 8.3 …...
测试直播postman+Jenkins所学
接口自动化 什么是接口?本质上就是一个url,用于提供数据。后台程序提供一种数据地址,接口的数据一般是从数据库中查出来的。 postman自动化实操: 一般来说公司会给接口文档,如果没有,通过拦截,…...
5人3小时复刻Manus?开源OpenManus项目全解剖,我的DeepSeek股票报告这样诞生
大家好,我是大 F,深耕AI算法十余年,互联网大厂技术岗。分享AI算法干货、技术心得。 更多文章可关注《大模型理论和实战》、《DeepSeek技术解析和实战》,一起探索技术的无限可能! OpenManus是什么 1. 项目背景 OpenManus 是由 MetaGPT 核心团队仅用 3 小时复刻而成的开源…...
【javaEE】多线程(基础)
1.❤️❤️前言~🥳🎉🎉🎉 Hello, Hello~ 亲爱的朋友们👋👋,这里是E绵绵呀✍️✍️。 如果你喜欢这篇文章,请别吝啬你的点赞❤️❤️和收藏📖📖。如果你对我的…...
0.大模型开发知识点需求综述
文章目录 一、机器学习与深度学习基础二、自然语言处理(NLP)基础三、大模型架构四、训练优化技术五、数据处理与预处理六、分布式训练与并行化策略七、微调方法与参数高效微调八、训练框架、工具与自动化流程九、评估与部署十、前沿技术与未来趋势 已更新…...
WHAT - 前端阻塞场景梳理
目录 前言场景梳理1. JavaScript 执行阻塞主线程场景优化思路具体代码示例1. 长时间运行的同步 JavaScript 代码2. 过多的主线程任务(如频繁的 setTimeout/setInterval)3. 未优化的第三方库或框架初始化逻辑总结 2. 样式计算与布局(Layout&am…...
微前端框架 Qiankun 的应用及问题分析
一、Qiankun 的核心应用场景与优势 多技术栈共存与灵活集成 Qiankun 支持主应用与子应用使用不同技术栈(如 Vue、React、Angular 等),通过 HTML Entry 方式接入子应用,无需深度改造子应用即可实现集成,降低了技术迁移成…...
bash: uwsgi: 未找到命令
针对你提出的“bash: uwsgi: command not found”问题,以下是一些可能的解决方案,请按照步骤逐一排查: 1、检查uwsgi命令是否正确: 确保你输入的命令是uwsgi,而不是uWSGI或其他变体。 2、确认uwsgi是否已安装&…...
HAL库,配置adc基本流程
1. 初始化阶段---cubemx (1) GPIO初始化 函数:HAL_GPIO_Init() 作用:配置ADC引脚为模拟输入模式。 代码示例: // 使能GPIOA时钟 __HAL_RCC_GPIOA_CLK_ENABLE();// 配置PA1为模拟输入 GPIO_InitTypeDef GPIO_InitStruct {0}; GPIO_InitStr…...
【Unity】 HTFramework框架(六十一)Project窗口文件夹锁定器
更新日期:2025年3月7日。 Github源码:[点我获取源码] Gitee源码:[点我获取源码] 索引 Project窗口文件夹锁定器框架文件夹锁定自定义文件夹锁定限制条件 Project窗口文件夹锁定器 在Project窗口中,文件夹锁定器能够为任何文件夹加…...
网络安全技术整体架构 一个中心三重防护
网络安全技术整体架构:一个中心三重防护 在信息技术飞速发展的今天,网络安全的重要性日益凸显。为了保护信息系统不受各种安全威胁的侵害,网络安全技术整体架构应运而生。本文将详细介绍“一个中心三重防护”的概念,并结合代码示…...
《AJAX:前端异步交互的魔法指南》
什么是AJAX AJAX(Asynchronous JavaScript and XML,异步 JavaScript 和 XML) 是一种用于创建异步网页应用的技术,允许网页在不重新加载整个页面的情况下,与服务器交换数据并局部更新页面内容。尽管名称中包含 XML&…...
Elasticsearch 2025/3/7
高性能分布式搜索引擎。 数据库模糊搜索比较慢,但用搜索引擎快多了。 下面是一些搜索引擎排名 Lucene是一个Java语言的搜索引擎类库(一个工具包),apache公司的顶级项目。 优势:易扩展、高性能(基于倒排索引…...
LLM论文笔记 19: On Limitations of the Transformer Architecture
Arxiv日期:2024.2.26机构:Columbia University / Google 关键词 Transformer架构幻觉问题数学谜题 核心结论 1. Transformer 无法可靠地计算函数组合问题 2. Transformer 的计算能力受限于信息瓶颈 3. CoT 可以减少 Transformer 计算错误的概率&#x…...
那年周五放学
2025年3月7日,周五,天气晴,脑子一瞬间闪过02-05年中学期间某个周五下午,17:00即将放学的场景,那种激动,那种说不上的欣喜感,放学后,先走一段316国道,再走一段襄渝铁路&am…...
002-SpringCloud-OpenFeign(远程调用)
SpringCloud-OpenFeign 1.引入依赖2.编写一个远程调用接口3.测试 1.引入依赖 <dependency><groupId>org.springframework.cloud</groupId><artifactId>spring-cloud-starter-openfeign</artifactId> </dependency><dependencyManageme…...
SAP 顾问的五年职业规划
SAP 顾问的职业发展受到技术进步、企业需求变化和全球经济环境的影响,因此制定长远规划充满挑战。面对 SAP 产品路线图的不确定性,如向 S/4HANA 和 Business Technology Platform (BTP) 的转变,顾问必须具备灵活性,以保持竞争力和…...
Pandas使用stack和pivot实现数据透视
Pandas的stack和pivot实现数据透视 经过统计得到多维度指标数据非常常见的统计场景,指定多个维度,计算聚合后的指标 案例:统计得到“电影评分数据集”,每个月份的每个分数被评分多少次:(月份,分…...
图像生成-ICCV2019-SinGAN: Learning a Generative Model from a Single Natural Image
图像生成-ICCV2019-SinGAN: Learning a Generative Model from a Single Natural Image 文章目录 图像生成-ICCV2019-SinGAN: Learning a Generative Model from a Single Natural Image主要创新点模型架构图生成器生成器源码 判别器判别器源码 损失函数需要源码讲解的私信我 S…...
c++ 操作符重载详解与示例
c 操作符重载详解与示例 操作符重载详解一、基本规则二、必须作为成员函数重载的运算符1. 赋值运算符 2. 下标运算符 []3. 函数调用运算符 ()4. 成员访问运算符 ->5. 转型运算符 三、通常作为非成员函数重载的运算符1. 算术运算符 2. 输入/输出运算符 << >> 四、…...
在Spring Boot项目中分层架构
常见的分层架构包括以下几层: 1. Domain 层(领域层) 作用:领域层是业务逻辑的核心,包含与业务相关的实体类、枚举、值对象等。它是对业务领域的抽象,通常与数据库表结构直接映射。 主要组件: 实体类(Entity):与数据库表对应的Java类,通常使用JPA或MyBatis等ORM框架…...
upload-labs详解(1-12)文件上传分析
目录 uploa-labs-main upload-labs-main第一关 前端防御 绕过前端防御 禁用js Burpsuite抓包改包 upload-labs-main第二关 上传测试 错误类型 upload-labs-env upload-labs-env第三关 上传测试 查看源码 解决方法 重命名,上传 upload-labs-env第四关…...
无人机应用探索:玻纤增强复合材料的疲劳性能研究
随着无人机技术的快速发展,轻量化已成为其结构设计的核心需求。玻纤增强复合材料凭借高强度、低密度和优异的耐环境性能,成为无人机机身、旋翼支架等关键部件的理想选择。然而,无人机在服役过程中需应对复杂多变的环境:高空飞行时…...
计算机毕业设计Python+DeepSeek-R1大模型空气质量预测分析(源码+文档+PPT+讲解)
温馨提示:文末有 CSDN 平台官方提供的学长联系方式的名片! 温馨提示:文末有 CSDN 平台官方提供的学长联系方式的名片! 温馨提示:文末有 CSDN 平台官方提供的学长联系方式的名片! 作者简介:Java领…...
【渗透测试】基于时间的盲注(Time-Based Blind SQL Injection)
发生ERROR日志告警 查看系统日志如下: java.lang.IllegalArgumentException: Illegal character in query at index 203: https://api.weixin.qq.com/sns/jscode2session?access_token90_Vap5zo5UTJS4jbuvneMkyS1LHwHAgrofaX8bnIfW8EHXA71IRZwsqzJam9bo1m3zRcSrb…...
学习threejs,Animation、Core、CustomBlendingEquation、Renderer常量汇总
👨⚕️ 主页: gis分享者 👨⚕️ 感谢各位大佬 点赞👍 收藏⭐ 留言📝 加关注✅! 👨⚕️ 收录于专栏:threejs gis工程师 文章目录 一、🍀前言1.1 ☘️Animation常量汇总1.1.1 循…...
2、数据库的基础学习(中):分组查询、连接查询 有小例子
二、分组函数 功能:用作统计使用,又称为聚合函数或者统计函数或组函数 1、分类: sum 求和、avg 平均值、max最大值、min 最小值、count 计算个数 2、参数支持哪些类型 Sum\avg 一般处理数值型数据 max、min 可以数值型也可以字符型…...
Ubuntu搭建最简单WEB服务器
安装apache2 sudo apt install apache2 检查状态 $ sudo systemctl status apache2 ● apache2.service - The Apache HTTP ServerLoaded: loaded (/lib/systemd/system/apache2.service; enabled; vendor prese>Active: active (running) since Thu 2025-03-06 09:51:10…...
如何学习编程?
如何学习编程? 笔记来源:How To Study Programming The Lazy Way 声明:该博客内容来自链接,仅作为学习参考 写在前面的话: 大多数人关注的是编程语言本身,而不是解决问题和逻辑思维。不要试图记住语言本身…...
OpenCV计算摄影学(14)实现对比度保留去色(Contrast Preserving Decolorization)的函数decolor()
操作系统:ubuntu22.04 OpenCV版本:OpenCV4.9 IDE:Visual Studio Code 编程语言:C11 算法描述 将彩色图像转换为灰度图像。它是数字印刷、风格化的黑白照片渲染,以及许多单通道图像处理应用中的基本工具。 cv::decolor 是 OpenCV…...
K8s 1.27.1 实战系列(七)Deployment
一、Deployment介绍 Deployment负责创建和更新应用程序的实例,使Pod拥有多副本,自愈,扩缩容等能力。创建Deployment后,Kubernetes Master 将应用程序实例调度到集群中的各个节点上。如果托管实例的节点关闭或被删除,Deployment控制器会将该实例替换为群集中另一个节点上的…...
Python第十五课:机器学习入门 | 从猜想到预测
🎯 本节目标 理解机器学习两大核心范式(监督/无监督学习)掌握特征工程的核心方法论实现经典算法:线性回归与K-Means聚类开发实战项目:房价预测模型理解模型评估与调优基础 一、机器学习核心概念(学生与老师…...