”工欲善其事,必先利其器。“—孔子《论语.录灵公》
首页 > 编程 > 什么是尖叫建筑?

什么是尖叫建筑?

发布于2024-11-09
浏览:154

What is Screaming Architecture?

Screaming Architecture 是由著名软件开发人员和思想领袖 Robert C. Martin(通常被称为“Bob 叔叔”)提出的概念。这个术语可能听起来非常规,但它代表了软件设计中的一个强大原则,专注于使系统架构反映应用程序的主要关注点和用例。简而言之,您的软件架构应该“尖叫”其意图和目的。

在这份综合指南中,我们将探讨 Screaming Architecture 的基础知识、它与传统软件架构的对比、它在领域驱动设计中的重要性,以及如何在项目中实现此架构。我们还将介绍 Screaming Architecture 可以提高代码可读性、可维护性和长期可扩展性的实际示例和场景。

为什么要“尖叫”建筑?

尖叫架构背后的想法是代码库的主要结构应该立即传达其业务目的。这与传统架构形成鲜明对比,传统架构可能强调技术框架、工具或其他次要问题。在尖叫架构中,领域关注优先于实现细节。

鲍勃·马丁叔叔用一个类比来说明这一点:想象一下走到一座建筑物前,看看它的建筑。无需标志,您通常就能辨别出这是图书馆、学校还是办公室。这同样适用于软件架构。当您查看应用程序的文件夹结构和设计时,您应该立即了解它的用途。如果你正在构建一个会计系统,那么架构应该大喊“会计”,而不是“Django”、“Spring Boot”或“React”。

以框架为中心的架构的问题

在许多项目中,对技术框架的关注掩盖了业务或领域逻辑。您会发现如下文件结构:

controllers/

services/

repositories/

models/

虽然这些目录很有用,但它们描述了技术角色,而不是反映软件解决的核心问题。例如,此结构告诉您系统使用 MVC(模型-视图-控制器),但没有深入了解系统是否处理财务数据、用户管理或内容创建。

框架陷阱

过分强调框架会导致代码库中的业务逻辑被技术样板所掩盖。围绕框架约定构建的系统与这些框架紧密耦合。如果你想改变框架或技术栈,重构就成为一项主要工作。尖叫架构提倡保持领域逻辑干净且独立,因此框架的选择成为实现细节,而不是代码库的核心结构。

领域驱动设计 (DDD) 中的尖叫架构

领域驱动设计(DDD)和尖叫架构经常齐头并进。 DDD 是一种强调技术和领域专家之间协作的软件开发方法,它侧重于以与实际操作紧密结合的方式对核心业务逻辑进行建模。

在 Screaming Architecture 中,领域模型和业务逻辑位于应用程序的中心,而其他所有内容(框架、数据库、UI 和服务)都变得次要。关键思想是代码结构应该反映领域模型而不是技术实现细节。

以下是如何使用领域驱动原则以“尖叫”其意图的方式构建项目:

/src
    /accounting
        Ledger.cs
        Transaction.cs
        Account.cs
        TaxService.cs
    /sales
        Order.cs
        Invoice.cs
        Customer.cs
        DiscountPolicy.cs

在此示例中,文件夹名称直接反映了业务问题:会计和销售。每个特定于域的类,如 Ledger、Transaction 和 Order,都放置在其相关的域上下文中。这种结构使系统的用途以及每个组件适合的位置立即清晰可见。

代码示例 1:简单的以领域为中心的结构

考虑一个处理订单和库存的电子商务应用程序。使用 Screaming Architecture,文件夹结构应该反映业务逻辑而不是技术角色:

/src
    /orders
        Order.cs
        OrderService.cs
        OrderRepository.cs
    /inventory
        InventoryItem.cs
        InventoryService.cs
        InventoryRepository.cs

这是订单上下文中的基本代码示例:

public class Order
{
    public Guid Id { get; set; }
    public DateTime OrderDate { get; set; }
    public List Items { get; set; }
    public decimal TotalAmount { get; set; }

    public Order(List items)
    {
        Id = Guid.NewGuid();
        OrderDate = DateTime.Now;
        Items = items;
        TotalAmount = CalculateTotal(items);
    }

    private decimal CalculateTotal(List items)
    {
        return items.Sum(item => item.Price * item.Quantity);
    }
}

public class OrderItem
{
    public string ProductName { get; set; }
    public decimal Price { get; set; }
    public int Quantity { get; set; }
}

在此代码中,域概念(Order)位于前端和中心,OrderService 和 OrderRepository 等支持逻辑保存在单独的文件中。业务逻辑 (CalculateTotal) 是 Order 实体的一部分,而不是隐藏在服务或控制器中。

避免技术干扰

框架和库对于软件开发至关重要,但它们不应该决定您的业务逻辑的结构。 Screaming Architecture 提倡将 HTTP 控制器、持久层和数据库框架等技术细节推向外围。

这是一个对比传统架构和尖叫架构的例子:

传统建筑:

/src
    /controllers
        OrderController.cs
    /services
        OrderService.cs
    /repositories
        OrderRepository.cs
    /models
        Order.cs
        OrderItem.cs

虽然这在技术上是正确的,但它并没有告诉您该系统的用途。文件夹结构没有透露任何有关域的信息。这是一个电子商务系统吗?金融应用程序?如果不深入代码,就不可能知道。

尖叫建筑:

/src
    /orders
        OrderController.cs
        OrderService.cs
        OrderRepository.cs
        Order.cs
        OrderItem.cs
    /inventory
        InventoryController.cs
        InventoryService.cs
        InventoryRepository.cs
        InventoryItem.cs

此结构立即阐明了系统处理订单和库存。如果您将来添加更多域(例如客户、付款),它们将在架构中拥有专用位置。

清洁架构的作用

尖叫架构通常与鲍勃叔叔更广泛的清洁架构原则保持一致。清洁架构提倡关注点分离,重点关注确保业务规则独立于框架、UI 和数据库。尖叫架构更进一步,建议项目的结构应该揭示核心业务逻辑。

以下是清洁架构的快速回顾:

实体:核心业务对象和逻辑。

用例:特定于应用程序的业务规则。

接口:框架和外部系统的网关。

框架和驱动程序:UI、数据库和其他外部组件。

在清洁架构项目中,订单、客户和发票等领域概念是中心层的一部分。 ASP.NET Core、Django 或 Rails 等框架被归入外层,充当提供核心功能的机制。

代码示例 2:在 Screaming Architecture 中应用清洁架构

在尖叫架构中,您将以反映业务领域的方式构建用例和实体。让我们扩展我们的电子商务示例:

/src
    /orders
        CreateOrderUseCase.cs
        OrderRepository.cs
        Order.cs
        OrderItem.cs
    /inventory
        AddInventoryItemUseCase.cs
        InventoryRepository.cs
        InventoryItem.cs

这是创建订单的示例用例:

public class CreateOrderUseCase
{
    private readonly IOrderRepository _orderRepository;
    private readonly IInventoryService _inventoryService;

    public CreateOrderUseCase(IOrderRepository orderRepository, IInventoryService inventoryService)
    {
        _orderRepository = orderRepository;
        _inventoryService = inventoryService;
    }

    public Order Execute(List items)
    {
        // Ensure all items are available in inventory
        foreach (var item in items)
        {
            _inventoryService.CheckInventory(item.ProductName, item.Quantity);
        }

        var order = new Order(items);
        _orderRepository.Save(order);
        return order;
    }
}

在此示例中,CreateOrderUseCase 是域逻辑的一部分,并与 OrderRepository 和 InventoryService 交互,以满足创建订单的业务需求。

尖叫架构的好处

提高可读性:任何打开代码库的人都会立即了解系统的功能。

关注点分离:业务逻辑与技术细节保持隔离,使得以后更容易更改框架或技术。

可扩展性:随着系统的增长,域结构保持一致,允许轻松添加新功能和模块。

可维护性:当领域逻辑与外部依赖项和框架完全分离时,它更容易维护。

与框架无关:Screaming 架构允许业务逻辑在不同的技术堆栈之间保持可移植性,避免与任何特定框架的紧密耦合。

对尖叫建筑的批评

虽然尖叫建筑有很多好处,但它也并非没有批评:

感知复杂性:不熟悉领域驱动设计的开发人员可能会发现,对于小型应用程序来说,领域逻辑与技术细节的分离是不必要的或过于复杂。
2

。开销:在小型项目或简单的 CRUD 应用程序中,实现 Screaming Architecture 可能看起来有点矫枉过正。

学习曲线:对于习惯框架优先方法的团队来说,采用 Screaming 架构需要转变思维,这可能需要时间来内化。

何时使用尖叫架构

Screaming Architecture 在以下场景中特别有用:

领域驱动系统:具有复杂业务规则和领域逻辑的应用程序。

长期项目:系统预计会随着时间的推移而发展,其中可扩展性和可维护性至关重要。

跨平台开发:可以切换框架或平台的系统,使得业务逻辑的清晰分离至关重要。

结论

尖叫建筑不仅仅是一个朗朗上口的名字;这是一种主张使核心业务逻辑成为代码库中最突出的部分的哲学。通过关注领域概念而不是技术框架,开发人员可以构建从长远来看更具可读性、可维护性和可扩展性的系统。无论您是在开发简单的 Web 应用程序还是复杂的企业系统,采用 Screaming Architecture 都可以生成更清晰、更集中的代码,清楚地表达其目的。

要了解有关 Screaming Architecture 的更多信息,您可以查看一些参考资料和附加读物:

罗伯特·C·马丁的清洁架构

Eric Evans 的领域驱动设计

鲍勃叔叔关于整洁代码的博客

通过采用尖叫架构的原则,您可以创建不仅可以工作而且可以“尖叫”其意图的代码库。

版本声明 本文转载于:https://dev.to/nilebits/what-is-screaming-architecture-442o?1如有侵犯,请联系[email protected]删除
最新教程 更多>
  • 可以在纯CS中将多个粘性元素彼此堆叠在一起吗?
    可以在纯CS中将多个粘性元素彼此堆叠在一起吗?
    [2这里: https://webthemez.com/demo/sticky-multi-header-scroll/index.html </main> <section> { display:grid; grid-template-...
    编程 发布于2025-03-21
  • 如何从Google API中检索最新的jQuery库?
    如何从Google API中检索最新的jQuery库?
    从Google APIS 问题中提供的jQuery URL是版本1.2.6。对于检索最新版本,以前有一种使用特定版本编号的替代方法,它是使用以下语法:获取最新版本:未压缩)While these legacy URLs still remain in use, it is recommended ...
    编程 发布于2025-03-21
  • 对象拟合:IE和Edge中的封面失败,如何修复?
    对象拟合:IE和Edge中的封面失败,如何修复?
    To resolve this issue, we employ a clever CSS solution that solves the problem:position: absolute;top: 50%;left: 50%;transform: translate(-50%, -50%)...
    编程 发布于2025-03-21
  • 如何使用FormData()处理多个文件上传?
    如何使用FormData()处理多个文件上传?
    )处理多个文件输入时,通常需要处理多个文件上传时,通常是必要的。 The fd.append("fileToUpload[]", files[x]); method can be used for this purpose, allowing you to send multi...
    编程 发布于2025-03-21
  • HTML格式标签
    HTML格式标签
    HTML 格式化元素 **HTML Formatting is a process of formatting text for better look and feel. HTML provides us ability to format text without us...
    编程 发布于2025-03-21
  • 如何从Python中的字符串中删除表情符号:固定常见错误的初学者指南?
    如何从Python中的字符串中删除表情符号:固定常见错误的初学者指南?
    从python import codecs import codecs import codecs 导入 text = codecs.decode('这狗\ u0001f602'.encode('utf-8'),'utf-8') 印刷(文字)#带有...
    编程 发布于2025-03-21
  • 如何克服PHP的功能重新定义限制?
    如何克服PHP的功能重新定义限制?
    克服PHP的函数重新定义限制在PHP中,多次定义一个相同名称的函数是一个no-no。尝试这样做,如提供的代码段所示,将导致可怕的“不能重新列出”错误。 但是,PHP工具腰带中有一个隐藏的宝石:runkit扩展。它使您能够灵活地重新定义函数。 runkit_function_renction_re...
    编程 发布于2025-03-21
  • 如何干净地删除匿名JavaScript事件处理程序?
    如何干净地删除匿名JavaScript事件处理程序?
    删除匿名事件侦听器将匿名事件侦听器添加到元素中会提供灵活性和简单性,但是当要删除它们时,可以构成挑战,而无需替换元素本身就可以替换一个问题。 element? element.addeventlistener(event,function(){/在这里工作/},false); 要解决此问题,请考虑...
    编程 发布于2025-03-21
  • 如何检查对象是否具有Python中的特定属性?
    如何检查对象是否具有Python中的特定属性?
    方法来确定对象属性存在寻求一种方法来验证对象中特定属性的存在。考虑以下示例,其中尝试访问不确定属性会引起错误: >>> a = someClass() >>> A.property Trackback(最近的最新电话): 文件“ ”,第1行, AttributeError: SomeClass...
    编程 发布于2025-03-21
  • 为什么Microsoft Visual C ++无法正确实现两台模板的实例?
    为什么Microsoft Visual C ++无法正确实现两台模板的实例?
    The Mystery of "Broken" Two-Phase Template Instantiation in Microsoft Visual C Problem Statement:Users commonly express concerns that Micro...
    编程 发布于2025-03-21
  • 如何使用PHP将斑点(图像)正确插入MySQL?
    如何使用PHP将斑点(图像)正确插入MySQL?
    essue VALUES('$this->image_id','file_get_contents($tmp_image)')";This code builds a string in PHP, but the function call ...
    编程 发布于2025-03-21
  • Android如何向PHP服务器发送POST数据?
    Android如何向PHP服务器发送POST数据?
    在android apache httpclient(已弃用) httpclient httpclient = new defaulthttpclient(); httppost httppost = new httppost(“ http://www.yoursite.com/script.p...
    编程 发布于2025-03-21
  • 如何使用PHP从XML文件中有效地检索属性值?
    如何使用PHP从XML文件中有效地检索属性值?
    从php $xml = simplexml_load_file($file); foreach ($xml->Var[0]->attributes() as $attributeName => $attributeValue) { echo $attributeName,...
    编程 发布于2025-03-21
  • 为什么PYTZ最初显示出意外的时区偏移?
    为什么PYTZ最初显示出意外的时区偏移?
    与pytz 最初从pytz获得特定的偏移。例如,亚洲/hong_kong最初显示一个七个小时37分钟的偏移: 差异源利用本地化将时区分配给日期,使用了适当的时区名称和偏移量。但是,直接使用DateTime构造器分配时区不允许进行正确的调整。 example pytz.timezone(...
    编程 发布于2025-03-21
  • 如何从PHP中的数组中提取随机元素?
    如何从PHP中的数组中提取随机元素?
    从阵列中的随机选择,可以轻松从数组中获取随机项目。考虑以下数组:; 从此数组中检索一个随机项目,利用array_rand( array_rand()函数从数组返回一个随机键。通过将$项目数组索引使用此键,我们可以从数组中访问一个随机元素。这种方法为选择随机项目提供了一种直接且可靠的方法。
    编程 发布于2025-03-21

免责声明: 提供的所有资源部分来自互联网,如果有侵犯您的版权或其他权益,请说明详细缘由并提供版权或权益证明然后发到邮箱:[email protected] 我们会第一时间内为您处理。

Copyright© 2022 湘ICP备2022001581号-3