单一职责原则有什么用?

2024-01-14

我试图理解单一职责原则,但我很难理解这个概念。我正在阅读 Lucian-Paul Torje、Adrian Ianculescu、Kamalmeet Singh 所著的《Java 设计模式和最佳实践》一书。

在这本书中我正在阅读单一职责原则章节, 他们的汽车类别如下所示:

他们说Car既有Car逻辑又有数据库操作。将来,如果我们想更改数据库,那么我们需要更改数据库逻辑,并且可能还需要更改汽车逻辑。反之亦然...

解决方案是创建两个类,如下所示:

我的问题是,即使我们创建两个类,让我们考虑一下我们正在向 CAR 类添加一个名为“price”的新属性 [或者将属性“model”更改为“carModel”] 那么您不认为我们还需要更新CarDAO类比如改变SQL等等。

那么SRP在这里有什么用呢?


很好的问题。

首先,请记住,这是书中的一个简单示例。读者可以对此进行扩展并想象更复杂的场景。在所有这些场景中,进一步想象您不是团队中唯一的开发人员;相反,您在一个大型团队中工作,开发人员之间的沟通通常采取协商类接口的形式i.e.API、公共方法、公共属性、数据库模式。此外,您通常还需要担心回滚、向后兼容性以及同步发布和部署。

例如,假设您想要将数据库从 MySQL 交换到 PostgreSQL。通过SRP,您将重新实施CarDAO,更改使用的任何特定于方言的 SQL,并保留Car逻辑完整。但是,您可能需要进行一些小更改(可能是在配置中),以告诉 Car 使用新的 PostgreSQL DAO。一个合理的 DI 框架会让这一切变得简单。

在另一个示例中,假设您希望将 CarDAO 委托给另一个开发人员来与 memcached 集成,以便读取虽然最终一致,但速度很快。同样,该开发人员不需要了解任何有关商业逻辑 in Car。相反,他们只需要在 CRUD 方法后面进行操作CarDAO,并可能在中声明更多方法CarDAO具有不同一致性保证的API。

再举一个例子,假设您的团队雇用了一位专门研究合规法的数据库工程师。为了准备即将到来的 IPO,数据库工程师的任务是保留公司 35 个数据库中所有表的所有更改的审核日志。有了 SRP,我们勇敢的 DBA 就不必担心任何问题商业逻辑使用我们的任何桌子;相反,他们的突变跟踪魔法可以通过使用装饰器或其他方面编程技术巧妙地注入到 DAO 中。 (顺便说一句,这也可以在 SQL 接口的另一端完成。)

好吧,最后一个 - 假设现在团队中引入了一名系统工程师,其任务是跨 AWS 中的多个区域(数据中​​心)分片这些数据。该工程师可以进一步改进 SRP,添加一个组件,该组件的唯一作用是告诉我们每个 ID 的每个实体的归属区域。每次我们进行跨区域读取时,新组件都会碰撞一个计数器;每周,自动化工具都会将跨区域频繁读取的数据迁移到新的主区域,以减少延迟。

现在,让我们进一步发挥想象力,假设业务正在蓬勃发展 - 突然间,您正在为一家财富 500 强公司工作,该公司的多个部门横跨多个国家。财务部的业务分析师希望使用您的表格在其 IPO 后投资者报告中绘制汽车销售的季度增长情况。而不是让他们访问Car(因为用于报告的逻辑可能与用于准备在 Web UI 上呈现的数据的逻辑不同),您可以为以下内容创建一个只读接口:CarDAO包含精心策划的公共属性的简短列表,您现在必须跨部门边界维护这些属性。上帝禁止你必须重新命名这些属性之一:为 3 个月的日落计划和许多悲伤的仪表板和深夜升级做好准备。 (并且请不要让他们直接访问实际的 SQL 表,因为隐含的假设是整个表是the公共界面。)哎呀,我的伤疤可能会露出来。

一个推论是,如果您需要更改商业逻辑 in Car(比如说,添加一个方法来计算每辆特斯拉在令人尴尬的召回后的较低销售价格),你不会碰CarDAO, since if car.brand == 'Tesla; price = price * 0.6与数据访问无关。

补充阅读:CQRS https://martinfowler.com/bliki/CQRS.html

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

单一职责原则有什么用? 的相关文章

  • 什么是 PHP 中的呼叫转移和静态呼叫或后期静态绑定? [关闭]

    Closed 这个问题需要多问focused help closed questions 目前不接受答案 我从网站上获得了一个代码示例 但我很难理解输出 我正在分享代码 class A public static function foo
  • 将我的数据库类与项目中的其他类一起使用

    我有一个自己编写的自定义数据库类 还有一个用户类和一个站点类 MySQL 类有这样的方法 connect query clean fetch 用户等级 register login logout resetPass 站点类别 updateT
  • Delphi 有通用的“对象池”实现吗?

    我在寻找 Delphi 的数据库连接池实现时遇到了这个问题 对象池需要两个方法 get 从池中获取一个对象 如果池为空或者其大小尚未达到最大大小 这将创建一个新实例 此方法必须是线程安全的 以便两个线程不能同时获取一个对象同一时间 如果所有
  • 在 C++ 中类继承的情况下强制延迟方法解析

    考虑以下类结构 class foo public int fun cout lt lt in foo lt lt endl class bar class1 public foo public int fun cout lt lt in b
  • 名称中带有点的 php 对象属性

    我有 mysql 表 其中包含 操作 日期 操作 名称 等列 将该表数据作为对象获取后 mysqli gt fetch object 我得到这个 行的 print r stdClass Object id gt 2 operation da
  • 棋子层次结构设计:继承与类型字段

    我有一个片段的基类 class piece 和一个包含派生对象的数组 piece board 8 8 优点 通过虚拟函数进行简洁的设计 缺点是 如果我必须在板上找到一块或比较一块 我必须恢复到动态铸造 或 typeid 它很丑陋 并且在发出
  • 在 Javascript 构造函数中引用实例变量

    我试图通过执行以下操作来维护对象的状态 obj function this foo undefined this changeState function function this foo bar This is contrived bu
  • 使用 OOP 开发 WordPress 插件

    我是插件开发的新手 所以 凡是我说得不对的地方 请大家指正 我有一个网站需要一个具有以下需求的播放器插件 管理员可控的玩家注册表单 其中包含一些详细信息 显示所有注册玩家的列表页面 已注册的玩家可以删除并相互比较 在前端的 WordPres
  • 作为属性的类引用

    谷歌对于这类搜索毫无用处 因为你会得到数亿个结果 其中没有一个与特定问题相关 问题很简单 Delphi 中是否可以有类引用属性 如果是这样 怎么办 这是我尝试过的 type TMyObject class end TMyObjectClas
  • 使用组合来表示“is – a”关系时出现的问题

    我正在为人力资源系统开发系统 有会计员工和程序员员工 加入公司的第一个月 员工没有被赋予任何角色 一名员工可以同时担任会计师和程序员 我有一个由以下代码所示的设计 现在 我需要通过实现新功能来增强系统 解雇所有会计师 终止意味着将员工的状态
  • 在 Ext JS 构造函数中将项目推入数组会产生多个项目

    我有一个我定义的 Ext JS 类 在这个班级的constructor 我将一个文本字段推送到我的项目数组上 然后添加到我的测试字符串中 数组和字符串在类定义中都声明为空 但是 如果您尝试创建多个类实例 您将看到项目数组在每个实例之间共享
  • 如何在 PHP 中检查特定类型的对象

    我有一种方法 它接受 PDO 对象作为参数 允许用户使用现有连接 而不是打开新连接的方法 并节省资源 public static function databaseConnect pdo null 我知道is object 检查参数是否是一
  • F# 对于 OO 或命令式来说缺少什么? [关闭]

    就目前情况而言 这个问题不太适合我们的问答形式 我们希望答案得到事实 参考资料或专业知识的支持 但这个问题可能会引发辩论 争论 民意调查或扩展讨论 如果您觉得这个问题可以改进并可能重新开放 访问帮助中心 help reopen questi
  • Java中的DRY原则[关闭]

    Closed 这个问题需要细节或清晰度 help closed questions 目前不接受答案 我一直在读关于DRY https en wikipedia org wiki Don 27t repeat yourself原则 虽然看起来
  • 依赖注入系统中的事件朝哪个方向发展?

    上或下 我是一个非常注重视觉的人 我将我的应用程序视为一个层次结构 顶部是根 底部是叶子 我还了解到 在 DI 系统中 容器不知道其所包含对象的职责 功能 相反 所包含的对象知道它们的上下文 因为上下文 依赖项 被注入 UP 非 DI 方式
  • 列出某个类的所有对象

    好的 我的问题如下 我有一个用这个构造函数描述宠物的类 public function construct name type age 所以我想做的是制作一些宠物对象 然后我想打印这个类的所有对象的所有属性 这样它看起来像this http
  • 如何在 Angular 2 应用程序中从 TypeScript/JavaScript 中的字符串获取类?

    在我的应用程序中 我有这样的内容 user ts export class User 现在 我这样做 应用程序组件 ts callAnotherFunction User 如果我将类名作为字符串 即 我该如何做到这一点 User 如果可能的
  • 如何为抽象工厂创建的类设置特定属性?

    是否可以让具体工厂使用抽象工厂模式为其创建具有特定类型参数的具体类 或者由各自的具体工厂创建的不同具体类是否需要具有相同的字段 例如 在下图中 您将如何使用客户端 应用程序 给出的不同参数集来实例化 WinButton 和 OSXButto
  • 一个类应该验证自身还是创建另一个类来验证它?

    假设我有一堂课 class NavigationData float roll float pitch double latitude double longitude 如果我想创建一个方法 const bool validate cons
  • 多重继承争论 II:根据 Stroustrup 的说法 [已关闭]

    Closed 这个问题是基于意见的 help closed questions 目前不接受答案 我非常了解关于为什么接口继承优于多重继承的传统争论 这里已经有一篇文章 C 应该有多重继承吗 https stackoverflow com q

随机推荐

  • Python 字符串匹配

    如果一个字符串包含 SUBJECT123 如何确定字符串有subject在Python中 if subject in mystring lower do something
  • Redis 键中冒号的用途是什么

    我正在学习如何在我的项目中使用 Redis 我不明白的一件事是键名称中冒号的确切用途 我见过这样的键名 users bob color blue item bag 冒号是否可以将键分类并加快查找键的速度 如果是这样 您可以在命名键时使用多个
  • 仅使用 CSS 使相邻同级元素具有相同的宽度

    我提前表示抱歉 因为出于保密原因 我无法显示我正在处理的代码 图像 但我认为我可以很简单地解释它 我有一个 h1 充当我的网页标题的元素 该标题可以根据用户所在的特定页面的标题更改长度 因此它可以说 主页 也可以说 已保存的项目 等 长度各
  • 特定版本的 HTC DESIRE HD 中 SQLite 中缺少表

    我的应用程序在 asset 文件夹中有一个 SQLite 数据库 当用户启动我的应用程序时 将创建数据库和表 这适用于许多设备 Nexus One Htc Magic SGS X10 甚至 Htc Desire HD v2 2 我的应用程序
  • 大 O 时间复杂度中的指数分母(分数指数)从何而来?

    In algorithm descirptions I sometimes encounter time complexities that look like O n29 20 m7 3 I see where and numerator
  • 为什么 stanford corenlp 性别识别是不确定的?

    我有以下结果 正如您所看到的 名字 edward 有不同的结果 空和男性 好几个名字都发生过这种情况 edward Gender null james Gender MALE karla Gender null edward Gender
  • ps au | 的结果grep ssh 在 Node.js(使用spawn/pipe)与 shell 中的不同

    我正在研究节点流和子进程 所以我想用管道模拟下一个 shell 命令 ps au grep ssh 所以我写了下一个代码 var spawn require child process spawn var ps spawn ps au va
  • 当浏览器不支持 JavaScript 时,如何在单击链接后执行脚本?

    我正在尝试使用如下所示的 Facebook 转换代码 其中包括script 和 noscripts 标签
  • rac_signalForSelector:需要空实现

    我有一个实现 UICollectionViewDelegate 协议的类 我正在使用 rac signalForSelector 来注册选择 如下所示 self rac signalForSelector selector collecti
  • 自定义 MapHttpAttributeRoutes 以进行 Web Api 版本控制

    我正在实施 Web API 版本控制 如下所示Web API 版本控制 http www codeproject com Articles 741326 Introduction to Web API Versioning 我的控制器位于
  • 网站部署后需要强制刷新

    部署新版本的网站后 浏览器会从旧网页的缓存中加载所有内容 直到强制刷新完成 图像是旧的 cookie 是旧的 并且一些 AJAX 部分无法工作 部署后我应该如何继续为用户提供最新版本的页面 该网页是使用IIS7 的ASP Net网页 您可以
  • 可靠地检测用户是否在网页上使用鼠标?

    在我们的页面上 我们有一些长水平滚动的 iframe div 如果用户使用鼠标 他们可以使用滚动条滚动 我们希望他们能够选择 iframe 中的文本 但是 如果他们仅使用触摸 则滚动条会很麻烦 我想在整个事物上覆盖一个透明元素 以使他们能够
  • 如何在 PostgreSQL 中生成虚拟表来生成日期序列?

    我想生成一个日期列表 希望与另一个表连接 但我不知道要使用什么语法 类似于 SELECT dates date transactions account id transactions amount FROM as dates LEFT J
  • 订票系统:数据库访问问题

    我正在创建一个巴士票预订系统 我创建了一个名为 Traveler 的数据库和两个分别名为 Useriden 和 BusDB 的表 在 aspx cs 文件 注册页面 中 我正在检查重复的用户名 但它只是导航到下一页 我已经尝试了一切 但无法
  • Rails 中的composed_of - 何时使用它?

    什么时候应该使用 ActiveRecordcomposed of http apidock com rails ActiveRecord Aggregations ClassMethods composed of类方法 就我个人而言 我认为
  • 将Java程序运行到另一个程序中[重复]

    这个问题在这里已经有答案了 可能的重复 在java程序中执行另一个jar https stackoverflow com questions 1320476 execute another jar in a java program 我尝试
  • 是否可以以编程方式向场景添加行?

    我想在每个 SpecFlow 测试的开头添加相同的行 这一行指定了几个场景的列表 这些场景会随着时间的推移而改变 因此为每个测试维护这个列表是不可行的 例如 Given I have set my site theme to
  • 发送用户 ID 和 access_token

    我正在使用 React 前端在 ASP NET Core 2 1 应用程序中实现 Auth0 一旦用户进行身份验证 我就会得到access token and an id token 我的目的很明确access token是授予对我的 AP
  • Chef-solo 从 bash 脚本获取日志

    我正在通过 Chef 执行 shell 脚本 如下所示 execute Run postgres data migration do command home ubuntu build target infra base psql10 mi
  • 单一职责原则有什么用?

    我试图理解单一职责原则 但我很难理解这个概念 我正在阅读 Lucian Paul Torje Adrian Ianculescu Kamalmeet Singh 所著的 Java 设计模式和最佳实践 一书 在这本书中我正在阅读单一职责原则章