热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

快速概览软件架构风格:从历史到现代的全面总结与分析

每个软件项目都是不同的,但这并不意味着我们没有可以挑选的趋势。大泥球(BigBallofMud)当您不选择架构时,您最终会得到BigBallofMud

每个软件项目都是不同的,但这并不意味着我们没有可以挑选的趋势。

大泥球(Big Ball of Mud)

当您不选择架构时,您最终会得到 "Big Ball of Mud" 架构。泥泞的“大球”是指从一个没有计划如何构建他们的解决方案的团队中出现的非结构化的意大利面条式代码库。

ea8de2e38110c1b189f39137387c3f7c.png

Big Ball of Mud 的名字来源于无组织的耦合和调用模式的网络,这些模式代表了没有清晰架构的项目.

分层架构

最简单和最常见的架构风格之一是分层架构。代码按技术领域划分为离散层。这清楚地表明了代码所属的位置。

d6a5a1367dbbdb76be884958b898f26c.png

将代码库典型地分解为层
一个典型的规则是每一层都是封闭的,这意味着它只能被它旁边的层访问,这有助于防止耦合。每一层都可以作为一个整体部署在一起,也可以单独部署。通常,表示层(UI)和数据库将分别部署到代码的其余部分。

管道架构

b3e94b278a78dc2315de0e75c0446d78.png

流水线架构以两种类型的单元为中心;管道和过滤器:

过滤器是独立的无状态计算单元。他们应该专注于完成一项特定任务。过滤器可以生成、测试、转换或使用数据。

管道是过滤器之间的通信通道。它们是一种方式,用于从过滤器构建管道。

微内核架构(插件架构)

82e9c575f902cd9e3a19af0b87793503.png

微内核架构基于构建核心系统的想法,该系统包含系统完成其核心目的所需的最少功能。然后通过独立插件提供附加功能。想想 VS Code:基本应用程序非常有限,但通过添加插件,您可以使应用程序非常强大,并为您和您的项目量身定制。

通常,插件彼此隔离,因此彼此分离。他们也有独立的存储。插件可以是具有核心系统的单体的一部分,也可以单独部署。如果与插件的通信是通过某种标准化形式的消息(如 REST)完成的,则无需要求所有内容都使用相同的语言编写。

基于服务的架构

966a1b565c5ee03f6ef5c44fc7a10e0f.png

在此体系结构中,功能由基于域的服务划分。该服务可以共享数据库和用户界面,但彼此独立运行。

服务可以相互独立部署。可以为需求较高的服务提供更多资源或进行复制。还可能有一个 API 代理将请求从 UI 路由到正确的服务。

事件驱动架构

0567a18564479a8cfe7abf0e0e784f7f.png

事件驱动系统对用户界面或内部发生的事件做出“反应”。这种系统是由事件处理器构建的;小的独立计算单元。通过系统的事件流可以以两种不同的方式发生。

在代理拓扑下,没有对事件处理的集中控制。每个处理器都可以广播自己的事件以触发另一个处理器形成事件链。不需要管理特定事件处理器或执行步骤之间的连接。

4686263b0ad9e12f4ca7a33ccf2e5588.png

在中介拓扑下有一些中央控制。事件调解器控制调用哪些处理器以及事件进入系统时的顺序。可能有多个事件中介以并行或分层方式工作。

微服务架构

8804f50d21e52ca980ce4f16a0079d6b.png

微服务系统由许多可独立部署的、以领域为中心的服务构建而成。“微”来自于每个服务应该只专注于完成一项任务的事实。微服务通常使用 REST 之类的东西(可能通过 API 网关)进行通信,用不同的语言编写并具有独立的存储。

微服务通常是独立部署的。这是一个“云原生”架构,它使用弹性来提供性能和可用性。

Clean Architecture

  • 域层:实体+用例+网关协议

  • 数据层:网关实现+ API(网络)+数据库

  • 表示层:视图模型+视图+导航器+场景用例

3755626b543e027b1e73dea9137c27eb.png

510c3087bcd43e456c0bfdd8140d8c76.png


26d2c7c0a229ed60f2c366c949595408.png


f67bd3135d9e5ca9333ca4ccff3b38b5.png

实体

实体封装了企业范围内的关键业务规则。实体可以是带有方法的对象,也可以是一组数据结构和函数。只要实体可以被企业中的许多不同应用程序使用,都没有关系。-清洁架构:软件结构和设计手工艺指南(Robert C. Martin)

实体是简单的数据结构:

struct Repo {var id = 0var name = ""var fullname = ""var urlString = ""var starCount = 0var folkCount = 0var avatarURLString = ""
}

用例

用例层中的软件包含特定于应用程序的业务规则。它封装并实现了系统的所有用例。这些用例协调了进出实体的数据流,并指示这些实体使用其关键业务规则来实现用例的目标。-清洁架构:软件结构和设计手工艺指南(Robert C. Martin)

UseCases是执行一项特定操作的协议:

protocol GettingRepos {var repoGateway: RepoGatewayType { get }
}extension GettingRepos {func getRepos(_ dto: GetPageDto) -> Observable> {repoGateway.getRepos(dto)}
}

数据传输对象-DTO

DTO-在进程之间传送数据的对象,它还执行数据验证:

struct LoginDto: Dto {@Validated(.nonEmpty(message: "Please enter user name"))var username: String?@Validated(.nonEmpty(message: "Please enter password"))var password: String?var validatedProperties: [ValidatedProperty] {return [_username, _password]}init(username: String, password: String) {self.username = usernameself.password = password}init() { }static func validateUserName(_ username: String) -> Result {LoginDto()._username.isValid(value: username)}static func validatePassword(_ password: String) -> Result {LoginDto()._password.isValid(value: password)}
}

网关协议

通常,网关只是另一个抽象,它将隐藏实际的实现,类似于外观模式。它可以是数据存储(存储库模式),API网关等。例如数据库网关将具有满足应用程序需求的方法。但是,请勿尝试在此类网关后面隐藏复杂的业务规则。对数据库的所有查询都应相对简单一些,例如CRUD操作,当然也可以接受一些过滤。-来源

protocol RepoGatewayType {func getRepos(_ dto: GetPageDto) -> Observable>
}

注意:为简单起见,我们将网关协议和实现放在同一文件中。实际上,网关协议应该在域层,实现应该在数据层。

资料层

f97fe90406e0424e72816d1ecf8323fa.png

数据层包含网关实现和一个或多个数据存储。网关负责协调来自不同数据存储的数据。数据存储可以是远程或本地(例如,持久数据库)。数据层仅取决于域层。

参考资料

https://blog.devgenius.io/some-software-architecture-styles-fbb57f7716b9

https://www.nuomiphp.com/github/zh/5fdc441a99daa46c1242dceb.html


推荐阅读
  • 1:有如下一段程序:packagea.b.c;publicclassTest{privatestaticinti0;publicintgetNext(){return ... [详细]
  • 优化ListView性能
    本文深入探讨了如何通过多种技术手段优化ListView的性能,包括视图复用、ViewHolder模式、分批加载数据、图片优化及内存管理等。这些方法能够显著提升应用的响应速度和用户体验。 ... [详细]
  • golang常用库:配置文件解析库/管理工具viper使用
    golang常用库:配置文件解析库管理工具-viper使用-一、viper简介viper配置管理解析库,是由大神SteveFrancia开发,他在google领导着golang的 ... [详细]
  • Navicat Premium 15 安装指南及数据库连接配置
    本文详细介绍 Navicat Premium 15 的安装步骤及其对多种数据库(如 MySQL 和 Oracle)的支持,帮助用户顺利完成软件的安装与激活。 ... [详细]
  • 探讨如何通过编程技术实现100个并发连接,解决线程创建顺序问题,并提供高效的并发测试方案。 ... [详细]
  • Explore a common issue encountered when implementing an OAuth 1.0a API, specifically the inability to encode null objects and how to resolve it. ... [详细]
  • Java 中 Writer flush()方法,示例 ... [详细]
  • 本文介绍了如何使用 Spring Boot DevTools 实现应用程序在开发过程中自动重启。这一特性显著提高了开发效率,特别是在集成开发环境(IDE)中工作时,能够提供快速的反馈循环。默认情况下,DevTools 会监控类路径上的文件变化,并根据需要触发应用重启。 ... [详细]
  • Java 中的 BigDecimal pow()方法,示例 ... [详细]
  • PHP 5.2.5 安装与配置指南
    本文详细介绍了 PHP 5.2.5 的安装和配置步骤,帮助开发者解决常见的环境配置问题,特别是上传图片时遇到的错误。通过本教程,您可以顺利搭建并优化 PHP 运行环境。 ... [详细]
  • 本文介绍了Java并发库中的阻塞队列(BlockingQueue)及其典型应用场景。通过具体实例,展示了如何利用LinkedBlockingQueue实现线程间高效、安全的数据传递,并结合线程池和原子类优化性能。 ... [详细]
  • 数据管理权威指南:《DAMA-DMBOK2 数据管理知识体系》
    本书提供了全面的数据管理职能、术语和最佳实践方法的标准行业解释,构建了数据管理的总体框架,为数据管理的发展奠定了坚实的理论基础。适合各类数据管理专业人士和相关领域的从业人员。 ... [详细]
  • 深入理解Cookie与Session会话管理
    本文详细介绍了如何通过HTTP响应和请求处理浏览器的Cookie信息,以及如何创建、设置和管理Cookie。同时探讨了会话跟踪技术中的Session机制,解释其原理及应用场景。 ... [详细]
  • 数据库内核开发入门 | 搭建研发环境的初步指南
    本课程将带你从零开始,逐步掌握数据库内核开发的基础知识和实践技能,重点介绍如何搭建OceanBase的开发环境。 ... [详细]
  • 本文深入探讨 MyBatis 中动态 SQL 的使用方法,包括 if/where、trim 自定义字符串截取规则、choose 分支选择、封装查询和修改条件的 where/set 标签、批量处理的 foreach 标签以及内置参数和 bind 的用法。 ... [详细]
author-avatar
伤心怪人_234
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有