清洁架构:遥不可及的理想——给开发者的寓言

清洁架构:遥不可及的理想——给开发者的寓言

在西藏宁静的山区高处,一座古老寺院安静的大厅里,住着一位年轻的学徒。他致力于追求和谐——不仅在他自己内部,而且在他的编程技巧中。他的目标是创建一个完美的应用程序,一个能够体现清洁架构深刻原理的应用程序,就像山间溪流的清澈一样。但他意识到这条道路的艰辛,于是向一位可敬的明师寻求智慧。

徒弟谦卑地走近师父问道

—“噢,明智的师父,我已经构建了一个应用程序来管理购买。我的建筑干净纯粹吗?”

师父耐心地观察弟子,回答道:

—“向我展示你所创造的东西,我们将一起辨别真相。”

学徒展示了他的作品,其中数据库逻辑和用例流程交织在一起——业务逻辑与技术框架紧密地交织在一起,就像一张错综复杂的网中的线。

// app.tsimport sqlite3 from 'sqlite3';import { open, database } from 'sqlite';interface purchase {    id: number;    title: string;    cost: number;}async function initializedatabase(): promise {    const db = await open({        filename: ':memory:',        driver: sqlite3.database,    });    await db.exec(`    create table purchases (      id integer primary key,      title text,      cost real    )  `);    return db;}async function addpurchaseifcan(db: database, purchase: purchase): promise {    const { id, title, cost } = purchase;    const row = await db.get(        `select sum(cost) as totalcost from purchases where title = ?`,        [title]    );    const totalcost = row?.totalcost || 0;    const newtotalcost = totalcost + cost;    if (newtotalcost  {    const db = await initializedatabase();    await addpurchaseifcan(db, { id: 3, title: 'rice', cost: 2 });})();

大师思考代码后说道:

——“你的代码就像一条河流,目的的清水与实现的泥浆混合在一起。业务和技术问题本应分开进行,但它们已合而为一。为了在你的建筑中实现真正的纯粹,你必须将它们分开,就像天空与大地分开一样。”

道路上的第一步

听从师父的话,徒弟开始着手重构他的代码。他开始分离各层,在数据库和业务逻辑流之间划出明显的界限。他还引入了接口,使他的代码与依赖倒置原则保持一致,这是清洁架构的神圣教义之一。现在,他的应用不再依赖于具体的实现,而是依赖于思想的抽象。

// app.tsimport { initializedatabase } from './db/init';import { purchaserepository } from './db/purchaserepository';import { addpurchaseifcan } from './usecases/addpurchaseifcan';(async () => {  const db = await initializedatabase();  const purchaserepository = new purchaserepository(db);  await addpurchaseifcan(purchaserepository, { id: 3, title: 'rice', cost: 2 });})();
// usecases/addpurchaseifcan.tsimport { ipurchaserepository, purchase } from './ipurchaserepository';export async function addpurchaseifcan(  purchaserepository: ipurchaserepository,  purchase: purchase): promise {  const { id, title, cost } = purchase;  const totalcost = await purchaserepository.gettotalcostbytitle(title);  const newtotalcost = totalcost + cost;  if (newtotalcost < 99999) {    await purchaserepository.add(purchase);    console.log('purchase added successfully.');  } else {    console.log('total cost exceeds 99999.');  }}
// usecases/ipurchaserepository.tsexport interface ipurchaserepository {  add(purchase: purchase): promise;  gettotalcostbytitle(title: string): promise;}export interface purchase {  id: number;  title: string;  cost: number;}
// db/init.tsimport sqlite3 from 'sqlite3';import { open, database } from 'sqlite';export async function initializedatabase(): promise {  const db = await open({    filename: ':memory:',    driver: sqlite3.database,  });  await db.exec(`    create table purchases (      id integer primary key,      title text,      cost real    )  `);  return db;}
// db/purchaserepository.tsimport { database } from 'sqlite';import { ipurchaserepository, purchase } from 'usecases/ipurchaserepository';export class purchaserepository implements ipurchaserepository {  private db: database;  constructor(db: database) {    this.db = db;  }  async add(purchase: purchase): promise {    const { id, title, cost } = purchase;    await this.db.run(      `insert into purchases (id, title, cost) values (?, ?, ?)`,      [id, title, cost]    );    return purchase;  }  async gettotalcostbytitle(title: string): promise {    const row = await this.db.get(      `select sum(cost) as totalcost from purchases where title = ?`,      [title]    );    const totalcost = row?.totalcost || 0;    return totalcost;  }}

徒弟回到师傅并问道:

——“我已经为我的存储库分离了各层并使用了接口。我的建筑现在干净了吗?”

大师再次检查代码,回复:

——“你们已经向前迈出了一步,但总成本的计算仍然停留在基础设施上,它不属于那里。这不是这种智慧应该存在的地方。总成本的知识属于商业领域,而不是地球上的工具。将其移至用例中,过程的智慧可以保持其纯粹性。”

分离的教训

有了这种洞察力,学徒意识到总成本的计算是业务逻辑的一部分。他再次重构了代码,将逻辑转移到用例中,使业务问题不受技术基础设施的影响。

// usecases/ipurchaserepository.tsexport interface ipurchaserepository {  add(purchase: purchase): promise;-  gettotalcostbytitle(title: string): promise;+  getpurchasesbytitle(title: string): promise;}...
// usecases/addpurchaseifcan.tsimport { ipurchaserepository, purchase } from './ipurchaserepository';export async function addpurchaseifcan(  purchaserepository: ipurchaserepository,  purchasedata: purchase,  limit: number): promise {  const { id, title, cost } = purchasedata;  const purchases = await purchaserepository.getpurchasesbytitle(title);  let totalcost = 0;  for (const purchase of purchases) {    totalcost += purchase.cost;  }  const newtotalcost = totalcost + cost;  if (newtotalcost >= limit) {    console.log(`total cost exceeds ${limit}.`);  } else {    await purchaserepository.add(purchasedata);    console.log('purchase added successfully.');  }}
// db/purchaserepository.tsimport { database } from 'sqlite';import { ipurchaserepository } from './ipurchaserepository';export class purchaserepository implements ipurchaserepository {  ...  async getpurchasesbytitle(title: string): promise {    const rows = await this.db.all(      `select * from purchases where title = ?`,      [title]    );    return rows.map((row) => ({      id: row.id,      title: row.title,      cost: row.cost,    }));  }}

再次回到师父身边,问道:

—“我已将总成本计算转移到用例中,并将业务逻辑与基础设施分开。我的建筑现在纯净了吗?”

师父带着温柔的微笑回答:

——“你已经取得了很大的进步,但要小心——就像山风带来冬天的寒冷一样,你的计算可能会带来隐藏的错误。 javascript 的算术就像新手的思维一样,在处理大数或小数时可能会不精确。”

与无常的相遇

学徒明白javascript中浮点运算的缺陷可能会导致微妙但危险的错误。他修改了代码,转向更可靠的工具,一个专为精确计算而设计的库,寻求工作的清晰度。

// usecases/addpurchaseifcan.ts+ import decimal from 'decimal.js';import { ipurchaserepository, purchase } from './ipurchaserepository';export async function addpurchaseifcan(  purchaserepository: ipurchaserepository,  purchasedata: purchase,  limit: number): promise {  const { id, title, cost } = purchasedata;  const purchases = await purchaserepository.getpurchasesbytitle(title);  let totalcost = new decimal(0);  for (const purchase of purchases) {-    totalcost += purchase.cost;+    totalcost = totalcost.plus(purchase.cost);  }- const newtotalcost = totalcost + cost;+ const newtotalcost = totalcost.plus(cost);- if (newtotalcost >= limit) {+ if (newtotalcost.greaterthanorequalto(limit)) {    console.log(`total cost exceeds ${limit}.`);  } else {    await purchaserepository.add(purchasedata);    console.log('purchase added successfully.');  }}

他再次问师父:

—“我已经改进了我的计算,使用了更好的工具来避免错误。我的建筑现在达到纯粹了吗?”

上人目光坚定如山,答道:

——“你做得很好,但你的架构仍然受到束缚。您的业​​务逻辑现在取决于这个新工具decimal.js的详细信息。如果有一天你需要改变这个工具,你的逻辑基础就会动摇。真正的纯洁是摆脱这种束缚。”

依赖倒置的智慧

认识到大师话语的深度,学徒试图将他的代码从这种执着中解放出来。他抽象了算术运算,颠倒了依赖关系,这样他的业务逻辑就不再依赖于任何一种工具。

// usecases/calculator.tsexport abstract class calculator {  abstract create(a: number): calculator;  abstract add(b: calculator | number): calculator;  abstract greaterthanorequal(b: calculator | number): boolean;}
// usecases/addpurchaseifcan.ts+ import { calculator } from 'usecases/calculator';- import decimal from 'decimal.js';import { ipurchaserepository, purchase } from './ipurchaserepository';
// decimalcalculator.tsimport decimal from 'decimal.js';import { calculator } from 'usecases/calculator.ts';export class decimalcalculator extends calculator {  private value: decimal;  constructor(value: number | decimal) {    super();    this.value = new decimal(value);  }  create(a: number): calculator {    return new decimalcalculator(a);  }  add(b: calculator | number): calculator {    return new decimalcalculator(this.value.plus(b.value));  }  greaterthanorequal(b: calculator | number): boolean {    return this.value.greaterthanorequalto(b.value);  }}
// useCases/addPurchaseIfCan.tsimport { Calculator } from 'useCases/calculator';import { IPurchaseRepository, Purchase } from './IPurchaseRepository';export class addPurchaseIfCan {  private purchaseRepository: IPurchaseRepository;  private calculator: Calculator;  private limit: string;  constructor(    purchaseRepository: IPurchaseRepository,    calculator: Calculator,    limit: number  ) {    this.purchaseRepository = purchaseRepository;    this.calculator = calculator;    this.limit = limit.toString();  }  async execute(purchaseData: Purchase): Promise {    const { id, title, cost } = purchaseData;    const purchases = await this.purchaseRepository.getPurchasesByTitle(title);    let totalCost = this.calculator.create(0);    for (const purchase of purchases) {      totalCost.add(purchase.cost);    }    totalCost = totalCost.add(cost);    if (totalCost.greaterThanOrEqual(this.limit)) {      console.log(`Total cost exceeds ${limit}.`);    } else {      await this.purchaseRepository.add({        id,        title,        cost: parseFloat(cost.toString()),      });      console.log('Purchase added successfully.');    }  }}

最后一次回到师父身边,他问道:

——“我使用依赖倒置抽象了我的操作,确保我的业务逻辑不再与实现绑定。我的建筑现在真的干净了吗?”

师父开示:

——“这条路上你已经走得很远了。但请记住,即使您努力追求纯度,您的用例仍然取决于编写它们的语言。您现在使用的是 javascripttypescript,但有一天这些工具可能会消失。当那一天到来时,你会用新的语言重建一切吗?”

拥抱不完美

学徒对此感到困惑,问道:

——“大师,如果我的用例总是与编写它们的语言联系在一起,我怎样才能在我的架构中实现完美的整洁?”

师父带着理解的柔和微笑回答:

——“就像鸟儿无法离开天空一样,建筑也不能完全脱离其创作的工具。真正的独立是一个崇高的梦想,但却是遥不可及的。然而,对它的追求会给你的建筑带来和谐。 清洁架构的目标不是摆脱所有依赖性,而是创建一个轻松应对变化的系统,并将商业智慧与地球运作分开。理解这种平衡是获得真正智慧的关键。”

学徒,感受到他内心的理解黎明之光,说道:

——“谢谢师父。现在我发现完美不是孤立的,而是责任和目标的和谐。”

师父从座位上站起来,回答:

——“放心吧,学徒。你的旅程才刚刚开始,但你已经找到了自己的路。”

结语

随着时间的流逝,学徒注意到他的应用程序开始变慢。他很困惑,想知道一个曾经运行得如此顺利的系统现在如何在执行任务时陷入困境。

很明显,问题不在于代码的大小不断增长,而在于总成本计算是在数据库外部进行的。该应用程序花费了大量的精力来传输大量数据,只是为了执行本可以在源头完成的任务。如果计算是在数据库内完成的,则无需在层之间发送数千条记录,并且性能仍将保持强劲。

徒弟想向师父询问此事,但师父已经消失,问题一直没有答案。

望着寂静的寺院,徒弟拿起一本新书,微笑着说道:

—“看来我的启蒙之路给我带来了新的挑战 – 性能优化的艺术。”

以上就是清洁架构:遥不可及的理想——给开发者的寓言的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1493240.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
Shadcn CLI如何使用错误常量来提高代码可读性
上一篇 2025年12月19日 15:31:36
清晰函数名称的力量:干净的代码必不可少
下一篇 2025年12月19日 15:31:40

相关推荐

  • Golang使用Protobuf定义接口与消息格式

    Protobuf通过字段编号实现兼容性,新增字段可忽略、删除字段可保留编号,确保新旧版本互操作,支持服务独立演进。 在Golang项目中,利用Protobuf定义接口和消息格式,本质上是为服务间通信构建了一套高效、类型安全且跨语言的契约。它让数据结构清晰可见,RPC调用标准化,极大地简化了分布式系统…

    2026年5月10日
    000
  • HTML文档如何工作?如何编辑HTML格式文件?

    HTML文档如何工作?如何编辑HTML格式文件?HTML文档如何工作?如何编辑HTML格式文件?HTML文档如何工作?如何编辑HTML格式文件?HTML文档如何工作?如何编辑HTML格式文件?

    浏览器解析和渲染html的过程包括:1. 解析html构建dom树;2. 结合css构建渲染树;3. 布局计算元素位置;4. 绘制像素到屏幕。编辑html可使用记事本、vs code、sublime text等文本或代码编辑器,其中vs code因语法高亮、自动补全和插件生态成为主流选择。标准htm…

    2026年5月10日 用户投稿
    100
  • php代码如何操作JSON数据_php代码解析和生成JSON的方法

    答案:PHP中处理JSON需使用json_encode()和json_decode()函数。1、将数组转为JSON字符串时,用json_encode()并检查返回值是否为false;2、解析JSON字符串时,调用json_decode()并设第二参数为true返回数组,false则返回对象;3、处理…

    2026年5月10日
    000
  • JS注解怎么和TypeScript结合_ JS注解在TypeScript环境下的应用

    TypeScript 支持通过配置 allowJs 和 checkJs 在 JavaScript 文件中识别 JSDoc 注解并进行类型检查,可在混合项目中提升类型安全;常见用法包括 @type、@param、@returns 和 @typedef,能为变量、函数参数等提供类型信息,支持与 .ts …

    2026年5月10日
    000
  • 解决Python脚本中相对路径文件找不到的常见问题与策略

    本文旨在解决python脚本中因相对路径处理不当导致的文件找不到错误,尤其是在项目迁移后。文章将深入探讨python中相对路径的工作原理、当前工作目录(cwd)的影响,并提供使用`os.getcwd()`诊断问题以及利用`os.path.dirname(__file__)`结合`os.path.jo…

    2026年5月10日
    000
  • C++内存检测工具 Valgrind使用实践指南

    Valgrind是一款主要用于Linux和macOS的内存调试工具,可检测内存泄漏、越界访问、未初始化内存使用等问题,通过memcheck工具结合–leak-check=full、–track-origins=yes等选项进行详细分析,需编译时添加-g选项以支持调试信息,虽然…

    2026年5月10日
    000
  • Golang如何提升TCP长连接处理效率_Golang TCP长连接处理性能优化实践详解

    答案:通过非阻塞I/O、单Goroutine双工模型、sync.Pool对象复用、TCP_NODELAY优化及高效心跳管理,结合系统调优,可显著提升Golang百万级TCP长连接处理效率。 在高并发网络服务场景中,TCP长连接的处理效率直接影响系统的吞吐能力和资源消耗。Golang凭借其轻量级Gor…

    2026年5月10日
    000
  • Go语言:检查预编译库的构建版本与平台信息

    本文详细介绍了如何利用go语言内置的`go tool pack`工具,从预编译的go静态库(`.a`文件)中提取其构建信息,包括go编译器版本、操作系统和cpu架构。当`go build`因库版本不匹配而失败时,此方法能帮助开发者准确诊断问题,确保构建环境与库的兼容性。 在Go语言的开发实践中,我们…

    2026年5月10日
    000
  • TypeScript函数体中如何高效判断参数类型?

    typescript 函数体中判断参数类型的技巧 typescript 中,我们可以定义接口来表示不同的数据类型。在本文中,我们将探讨如何在函数体中判断参数的类型,从而实现类型收窄,进行更精细的类型检查。 使用谓词函数 一种方法是编写谓词函数来手动检查类型。谓词函数返回的是 value is som…

    2026年5月10日
    000
  • JS如何实现策略模式

    策略模式通过封装算法使其可互换,JavaScript中利用函数作为一等公民实现,适用于表单验证等场景,结合工厂模式提升灵活性,但应避免过度设计。 策略模式的核心在于定义一系列算法,并将每一个算法封装起来,使它们可以相互替换。这使得算法可以在不影响客户端的情况下发生变化。在JS中,这可以通过函数作为一…

    2026年5月10日
    000
  • Svelte中正确导入数据与组件:避免常见误区

    在svelte开发中,理解如何正确导入数据和组件至关重要。svelte文件定义的是组件而非普通javascript模块,若需共享纯数据,应使用`.js`文件进行导出。本文将详细阐述svelte的导入机制,并通过示例代码展示如何区分导入数据与渲染组件,从而避免常见的导入错误,确保项目结构清晰且功能正确…

    2026年5月10日
    300
  • 如何通过GitHub API高效获取超过100个用户列表(分页教程)

    本教程旨在解决使用GitHub API获取用户列表时遇到的默认100个用户限制问题。我们将详细介绍两种主要的分页策略:利用Octokit库内置的paginate方法实现自动化分页,以及手动实现基于since参数的循环分页逻辑。文章将提供清晰的代码示例,并强调在不同场景下选择合适方法的注意事项,特别是…

    2026年5月10日
    100
  • 前端构建优化:利用常量折叠提升应用性能

    本文深入探讨了一种在构建阶段执行部分源代码以进行优化的技术——常量折叠(Constant Folding)。通过在编译时预计算表达式并替换为最终结果,该技术显著减少了运行时开销,提升了应用性能。文章将详细解释其工作原理、优势,并探讨其在现代前端构建工具中的应用与配置,旨在帮助开发者实现更高效的代码优…

    2026年5月10日
    000
  • 动态语言中静态类型的讽刺

    c++kquote>您也可以在 medium 上阅读这篇文章。 当我们看到编程语言如何随着时间的推移而演变时,总是很有趣。 曾几何时,当我开始进入软件开发世界时,python、php 和 javascript 等动态语言因其灵活性和适合快速开发的简洁语法而受到赞赏。 然而,随着这些弱类型语言的…

    2026年5月10日
    000
  • 为什么 TypeScript 比 JavaScript 更好

    javascript 长期以来一直是 web 开发的基石,支持从小型脚本到大型应用程序的各种项目。然而,随着项目规模的扩大,javascript 的动态类型和缺乏结构性可能会成为开发的瓶颈。typescript 应运而生,它凭借静态类型检查和强大的工具集,迅速成为许多开发者构建可靠、可扩展应用程序的…

    2026年5月10日
    100
  • 深入解析Angular中循环计算与数组操作的常见陷阱及优化实践

    本文深入探讨了angular应用中处理循环计算和动态数组时常见的逻辑错误。通过一个租金计算器示例,我们分析了`for`循环中未能正确累加迭代值以及数组填充不当的问题,并提供了详细的解决方案,包括优化计算逻辑、正确使用数组`push`方法,以及遵循typescript和javascript的最佳实践,…

    2026年5月10日
    000
  • Python官网函数库的深入学习_Python官网标准库高级用法解析

    掌握Python标准库高级用法需深入functools、itertools、subprocess、pathlib和concurrent.futures模块:1. functools的@lru_cache可缓存递归结果提升性能;2. itertools提供product、groupby和cycle等工…

    2026年5月10日
    000
  • Angular Material Table 数据源的正确绑定与异步数据处理

    在 Angular 应用中,将异步获取的数据正确绑定到 Material Table 的 `MatTableDataSource` 是一个常见挑战。本文将深入探讨 `MatTableDataSource` 的初始化时机,特别是如何处理数据加载的异步性,确保表格能够实时、准确地渲染数据,并提供一个结构…

    2026年5月10日
    000
  • JS注解怎么标注只读属性_ JS对象只读属性的注解方法与使用

    使用JSDoc的@readonly可标注只读属性以提升开发体验,但需结合Object.defineProperty设置writable为false才能实现运行时保护,TypeScript的readonly关键字则能在编译阶段阻止修改,提供更强的类型检查。 在JavaScript中,给对象属性设置只读…

    2026年5月10日
    100
  • 微服务中的配置漂移如何防止?

    防止配置漂移需统一管理、版本控制和自动化;2. 使用配置中心集中存储配置,实现动态刷新与权限控制;3. 配置与代码分离并纳入Git,支持审计与CI/CD集成;4. 保持多环境配置结构一致,通过模板生成差异值;5. 容器化与IaC实现不可变基础设施,杜绝手动修改。 微服务架构中,配置漂移指的是不同环境…

    2026年5月10日
    200

发表回复

登录后才能评论
关注微信