基于「骑手外卖系统」串联7大设计原则

你说得对!这些设计原则听起来都很抽象、很“玄”,如果不是实际开发过系统,很难理解“到底为什么要这样设计”。

那我现在就用一个你能想象得很清楚的真实例子,帮你把这7个设计原则一一落地到具体情境里,你会一眼明白。


🧩 场景设定:你在开发一个「外卖系统」

像饿了么、美团那样的平台。我们假设你要设计一个“骑手配送子系统”。


🚴 用真实任务解释 7 个设计原则

1️⃣ 单一职责原则(SRP)

每个类只干一件事

✅ 你设计了两个类:

class Rider:def deliver_order(self): ...def update_gps(self): ...def withdraw_earnings(self): ...class PaymentService:def process_withdrawal(self): ...

📌 问题:如果 Rider 也管提现,那它责任太杂 → 违反 SRP
✅ 把提现相关内容抽成 PaymentService,每个类只管一件事。


2️⃣ 开闭原则(OCP)

增加新功能,别改老代码

用户下单时,默认是“普通配送”。

现在老板说:加个“帮我带瓶奶”的特殊配送类型。

✅ 好的,你做法是:

class DeliveryStrategy(ABC):def deliver(self): ...class NormalDelivery(DeliveryStrategy):def deliver(self): print("按原计划送")class ErrandDelivery(DeliveryStrategy):def deliver(self): print("顺便买瓶奶")# ✅ 原来用 NormalDelivery,现在只要新增一个类就行

📌 没动旧逻辑(开闭原则 ✅)


3️⃣ 接口隔离原则(ISP)

别强迫子类实现它用不到的方法

你定义了一个接口:

class RiderInterface:def deliver_food(self): ...def fly_drone(self): ...

📌 大部分骑手不会飞无人机,但你却强迫所有人实现这个接口 → 违反 ISP

✅ 正确做法:

class FoodDelivery:def deliver(): ...class DroneDelivery:def fly(): ...

谁会什么功能,就实现什么接口,别搞全能型。


4️⃣ 里氏替换原则(LSP)

子类必须能安全地替代父类

你有个函数:

def assign_order(rider: Rider):rider.deliver()

但你有一个子类 LazyRider

class LazyRider(Rider):def deliver(self): raise Exception("我今天不想送")

📌 语法没错,运行会炸。这说明 LazyRider 不能真正替代 Rider → 违反 LSP

✅ 正确做法:只有真正能送外卖的才能继承 Rider


5️⃣ 依赖倒置原则(DIP)

不依赖细节,只依赖“能力”

配送逻辑里你不直接写:

rider = AutoRider()  # ❌死绑定某类
rider.deliver()

✅ 而是:

def dispatch(rider: DeliveryInterface):rider.deliver()

📌 这样你能传入自动骑手、普通骑手、临时工……
程序只关心“他能送”这个接口,而不是他是谁 → 依赖倒置 ✅


6️⃣ 合成复用原则(CRP)

能组合就别继承

你一开始写:

class SuperRider(DeliveryService):def deliver(): ...def repair_bike(): ...

📌 不要为了“多功能”就继承,而应该组合:

class RepairService:def repair(): ...class SuperRider:def __init__(self):self.repair_service = RepairService()def fix_bike(self):self.repair_service.repair()

✅ 这样如果后续要换维修流程,不改 SuperRider → 组合优于继承


7️⃣ 迪米特法则(LoD)

不该知道的事就别打听

你有一个骑手类:

class Rider:def get_bike().get_engine().check_status()

📌 你“越级”访问了三层 → 太过深入别人的内部结构
✅ 正确:

rider.check_bike_status()

📌 你只知道“骑手能查车况”,不用知道车具体怎么查 → 降耦合、守住职责边界


🧠 总结表格:骑手外卖系统中 7 个设计原则怎么落地?

原则在外卖系统中的例子守住了什么?
✅ SRP 单一职责把提现从 Rider 拆出去让每个类更专注
✅ OCP 开闭原则增加新配送方式,不动旧代码提高扩展能力
✅ ISP 接口隔离不强制 Rider 实现“飞无人机”让接口更贴合实际
✅ LSP 里氏替换LazyRider 会炸 → 不合适保证多态安全
✅ DIP 依赖倒置用接口注入 rider,不依赖具体类解耦调用与实现
✅ CRP 合成复用用 RepairService 而不是继承优先组合、灵活替换
✅ LoD 迪米特法则不越级访问 bike.engine降低耦合风险

如果你能记住这个「骑手外卖系统」的例子,以后看到这7个设计原则就不再抽象了,而是具体而清晰

你要我做一个【图文并茂的思维导图 + 表格模版】帮助你记住这整套系统吗?📘

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

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

相关文章

基于轻量化YOLO的车载设备实时视频目标检测方案

一、背景与需求 在智能汽车后装市场,设备的视觉感知能力日益成为差异化竞争点。传统后装产品主要关注车辆诊断和位置跟踪,但在智能辅助驾驶(ADAS)与车联网(V2X)大潮下,如果能在已有硬件平台上新增“视频级行人、车辆、交通标志检测”功能,不仅可提升安全预警,也能为后…

HTTPS协议:更安全的HTTP

目录 1. 前言 2. HTTP 与 HTTPS:安全的分水岭 2.1 HTTP 的安全隐患 2.2 HTTPS 的安全提升 3. HTTPS 的核心概念 3.1 加密三剑客:对称加密、非对称加密与哈希算法 3.2 SSL/TLS 握手过程:建立安全通道的关键步骤 3.3 数字证书&#xff…

使用 Go 和 Gorgonia 实现图像验证码识别系统

本文将介绍如何使用 Go 语言结合 Gorgonia 构建一个简单的图像验证码识别模型。Gorgonia 是一个专为机器学习打造的计算图库,在 Go 中支持自动微分与深度学习构建。 1. 项目依赖 首先安装 Go 语言环境和 Gorgonia: 登录后复制 go install gorgonia.org/g…

list的两种设计

1. 内存布局对比 (1) MSVC 的实现 cpp class _List_node {_List_node* _Next; // 指向下一个节点_List_node* _Prev; // 指向前一个节点_Value_type _Value; // 存储的数据 }; 特点: 每个节点包含两个指针和一个数据成员。 Debug 模式:可能添加迭代…

多多铃声 7.4| 拥有丰富的铃声曲库,满足不同用户的个性化需求,支持一键设置手机铃声

多多铃声是一款提供丰富铃声资源的应用程序,它拥有广泛的铃声曲库,涵盖各种风格和类型,能够满足不同用户的个性化需求。该应用程序支持分类浏览和热门榜单功能,让用户可以轻松找到当前最流行或自己感兴趣的铃声。此次分享的版本为…

Day04 新增套餐

###今天的任务主要是自主完成套餐管理的模块### 1.新增套餐 在前端页面接口中我们可以看到在新增套餐的时候需要选择添加到菜单中的菜品 因此我们需要设计一个接口可以通过根据分类id(category_id)来查询该分类下的菜品 1.1根据分类id查询分类下的菜…

数据赋能(208)——质量管理——及时性原则

概述 在数据处理、分析和应用过程中,数据及时性原则确保了数据在需要时能够迅速、准确地被获取、更新和传递,为决策和业务需求提供了时效性保障。能够反映当前的真实状况,为决策提供最新、最准确的信息支持。这种及时性不仅有助于企业快速响…

华为OD机试真题——告警抑制(2025A卷:100分)Java/python/JavaScript/C/C++/GO最佳实现

2025 A卷 100分 题型 本专栏内全部题目均提供Java、python、JavaScript、C、C、GO六种语言的最佳实现方式; 并且每种语言均涵盖详细的问题分析、解题思路、代码实现、代码详解、3个测试用例以及综合分析; 本文收录于专栏:《2025华为OD真题目录…

ASP.NET MVC​ 入门指南四

21. 高级路由配置 21.1 自定义路由约束 除了使用默认的路由约束,你还可以创建自定义路由约束。自定义路由约束允许你根据特定的业务逻辑来决定一个路由是否匹配。例如,创建一个只允许特定年份的路由约束: csharp public class YearRouteCo…

测试基础笔记第十八天

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 一、web自动化1.xpath定位1.属性定位2.属性与逻辑结合3.属性和层级结合 2.常见元素定位方法(面试题)3.常见元素定位失败原因4.cookie1.验证码…

(笔记)List

一、List的介绍和使用 1.List的介绍 1.1 list是可以在任意常数范围内插入和删除的序列式容器,并且该容器可以前后双向迭代。 1.2 list底层是双向链表结构,双向链表中每个元素都储存在互不相关的独立节点中,在节点中通过指针指向前其前一个…

重载和覆写有什么区别?

重载(Overload)和覆写(Override,也叫重写 )是面向对象编程中的重要概念,它们有以下区别: 定义 - 重载:在同一类中,允许存在多个方法名相同,但参数列表&#x…

flask 获取各种请求数据:GET form-data x-www-form-urlencoded JSON headers 上传文件

在 Flask 里,能使用多种方法获取不同类型的请求数据,下面详细介绍常见请求数据的获取方式。 获取查询字符串参数(GET 请求) 查询字符串参数一般在 URL 里,以 ?key1value1&key2value2 这种形式存在。可通过 requ…

人工智能助力工业制造:迈向智能制造的未来

在当今数字化转型的浪潮中,人工智能(AI)技术正逐渐成为推动工业制造领域变革的核心力量。智能制造作为工业 4.0 的重要组成部分,通过将 AI 技术与传统制造工艺深度融合,正在重塑整个生产流程,提高生产效率、…

【java八股文】深入浅出synchronized优化原理

🔍 开发者资源导航 🔍🏷️ 博客主页: 个人主页📚 专栏订阅: JavaEE全栈专栏 synchronized优化原理 synchronized即使悲观锁也是乐观锁,拥有自适应性。 jvm内部会统计每个锁的竞争激烈程度&…

生成式 AI 的重要性

在科技飞速发展的今天,我们正站在一个前所未有的变革节点上。生成式 AI,宛如一颗突然划破夜空的耀眼流星,以其强大的创造力和无限的可能性,迅速成为全球瞩目的焦点。它究竟有何等魔力,能在如此短的时间内引发如此巨大的轰动?这背后又隐藏着怎样的时代密码,等待着我们去解…

生成式 AI 的阐释

在科技浪潮的汹涌推动下,一个全新的时代正以前所未有的速度向我们奔来,生成式 AI 无疑是这股浪潮中最耀眼的浪尖。它究竟是什么?又将如何重塑我们的世界?这不仅是科技爱好者们热衷探讨的话题,更是关乎每一个人未来发展的重要命题。 生成式 AI,从本质上来说,是一种能够自…

C++ 中 virtual 的作用

文章目录 1. 用于虚继承2. 用于抽象基类3. 用于多态 C 的 virtual 关键字,常见有 3 种用途:虚继承、抽象基类和多态。 1. 用于虚继承 virtual 用于虚继承时,主要是为了解决菱形继承中的重复继承同名成员问题。使用形式如下 class Derived:…

软件测试52讲学习分享:深入理解单元测试

课程背景 最近我在学习极客时间的《软件测试52讲》课程,这是由腾讯TEG基础架构部T4级专家茹炳晟老师主讲的认证课程。作为数字化转型与人工智能(DTAI)产业人才基地建设中心的认证课程,内容非常专业实用。今天想和大家分享第3讲"什么是单元测试&…

Java SE(7)——类和对象(二)

1.包(package) 1.1 包的定义 在Java中,包是一种用于组织和管理类,接口和其他包的机制。主要作用是防止命名冲突,并提供一种访问控制机制 1.2 package关键字 package关键字的主要作用是声明当前类在哪个包里面。 当然,用户也可以…