《从零开始学架构》读书笔记(一)

目录

软件架构设计产生的历史背景

软件架构设计的目的

系统复杂度来源

追求高性能

一、单机高性能

二、集群的高性能

追求高可用

一、计算高可用

二、存储高可用

追求可扩展性

一、预测变化

二、应对变化

追求安全、低成本、规模

一、安全

二、低成本

三、规模

读书小结


前言

        作为后端开发攻城狮,学习并掌握一些架构的思想和技能是很有必要的。

        《从零开始学架构》的作者李运华,资深技术专家。这本书是他从业十多年,对架构设计的心得和方法论的总结,比较系统化也比较通俗易懂;正如他在书中开篇提到的,照着做,你也能成为架构师。这很接地气,也正是吸引我的地方。我打算按照读书的进度,记录和总结书中的主要内容,再结合我的理解,整理成几篇读书笔记,定时更新。

软件架构设计产生的历史背景

        软件架构概括成一句话:系统的组织形式、内部联系和逻辑。

        最早在60年代,就已经出现“软件架构”这个概念了,但真正流行却是从90年代开始的,这要从2次软件危机说起:

        第一次软件危机与结构化程序设计(60年代~70年代),这次危机的根源在于软件的“逻辑”变得非常复杂,这次危机催生了结构化程序设计方法结构化程序方法成为了 70年代软件开发的潮流。

        第二次软件危机与面向对象(80年代),这次危机主要体现在软件的“扩展”变得非常复杂,这次危机,促进了面向对象的发展,80年代开始面向对象成为主流的开发思想。

        90年代,人们对软件架构的研究发现:随着软件系统规模的增加,计算相关的算法和数据结构不再构成主要的设计问题;当系统由许多部分组成时,整个系统的组织,也就是所说的“软件架构”,导致了一系列新的设计问题。这样很好解释了在一些大规模的软件开发中,非常容易面临的问题,如系统规模庞大,内部耦合严重,续修改和扩展困难;系统逻辑复杂,出问题后很难排查和修复。

        70年代的危机主要是逻辑复杂问题,结构化编程可以解决这问题,创造了“模块”概念;80年代的危机主要是软件扩展问题,面向对象思想可以解决这问题,创造了“对象”;90年代的“软件架构”开始流行,创造了“组件”概念。我们可以看到,“模块”“对象”“组件”本质上都是对达到一定规模的软件进行拆分,差别只是在于随着软件的复杂度不断增加,拆分的粒度越来越粗,拆分的层次越来越高。

软件架构设计的目的

        软件架构设计的目的是为了解决软件系统复杂度带来的问题。为什么这么说,从前面的两次软件危机可以看到,随着系统规模、业务需求的不断增长,系统的复杂度(功能、性能、扩展、安全等等统称为复杂度)会不断上升,但上升到一定程度后,会导致现有系统难以继续开发维护。引进架构的设计,就是在系统开发之初,设计可行的架构方案,并在迭代过程,及早的发现可能出现的危机,并针对危机,对软件的架构进行调整优化。

系统复杂度来源

        既然架构设计的目的是解决系统复杂度带来的问题,那就要先了解导致系统复杂度增加的几个主要来源:

追求高性能

一、单机高性能

        单机的局限性明显,主要是通过多进程、多线程的方式,提高单机的的吞吐量。

二、集群的高性能

        单机的性能有限,必须采用机器集群的方式来达到高性能。例如,支付宝和微信这种规模的业务系统,后台系统的机器数量都是万台级别的。

1、任务分配

        任务分配的意思是指每台机器都可以处理完整的业务任务,不同的任务分配到不同的机器上执行。分配器可能是硬件网络设备,也可能是负载均衡软件(例如,Nginx、HAProxy)等,任务分配器需要增加分配算法。一般来说机器数量越多,集群整体的性能越强。

2、任务分解

        如果业务本身也越来越复杂,单纯只通过任务分配的方式来扩展性能,收益会越来越低。可以将其拆分为更多的组成部分,例如微信的后台架构。

        通过这种任务分解的方式,能够把原来大一统但复杂的业务系统,拆分成小而简单但需要多个系统配合的业务系统。为何通过任务分解就能够提升性能呢?

  • 简单的系统更加容易做到高性能

        系统的功能越简单,影响性能的点就越少,就更加容易进行有针对性的优化。而系统很复杂的情况下,首先是比较难以找到关键性能点,因为需要考虑和验证的点太多;其次是即使花费很大力气找到了,修改起来也不容易,因为可能将 A 关键性能点提升了,但却无意中将 B 点的性能降低了,整个系统的性能不但没有提升,还有可能会下降。

  • 可以针对单个任务进行扩展

        当各个逻辑任务分解到独立的子系统后,整个系统的性能瓶颈更加容易发现,而且发现后只需要针对有瓶颈的子系统进行性能优化或者提升,不需要改动整个系统,风险会小很多。

3、任务分解的误区

        既然将一个大一统的系统分解为多个子系统能够提升性能,那是不是划分得越细越好呢?其实不然,这样做性能不仅不会提升,反而还会下降,最主要的原因是如果系统拆分得太细,为了完成某个业务,系统间的调用次数会呈指数级别上升,而系统间的调用通道目前都是通过网络传输的方式,性能远比系统内的函数调用要低得多。因此,任务分解带来的性能收益是有一个度的,并不是任务分解越细越好。

追求高可用

        和之前高性能是一样的,都是通过增加更多机器来达到目的,但其实本质上是有根本区别的:高性能增加机器目的在于“扩展”处理性能;高可用增加机器目的在于“冗余”处理单元。

一、计算高可用

        这里的“计算”指的是业务的逻辑处理。计算有一个特点就是无论在哪台机器上进行计算,同样的算法和输入数据,产出的结果都是一样的,所以将计算从一台机器迁移到另外一台机器,对业务并没有什么影响。

二、存储高可用

        存储高可用的痛点,在于多个数据库或缓存服务器之间,数据不一定时时刻刻是一致的,可能存在某一个时刻数据不一致的问题,例如主从数据库的数据同步可能存在网络延迟。

        按照“数据 + 逻辑 = 业务”这个公式来套的话,数据不一致,即使逻辑一致,最后的业务表现就不一样了。所以存储高可用的难点不在于如何备份数据,而在于如何减少或者规避数据不一致对业务造成的影响

        著名的 CAP 定理,从理论上论证了存储高可用的复杂度。也就是说,存储高可用不可能同时满足“一致性、可用性、分区容错性”,最多满足其中两个。后续的读书笔记中详细展开说明吧。

追求可扩展性

        设计具备良好可扩展性的系统,有两个基本条件:正确预测变化完美封装变化。但要达成这两个条件,本身也是一件复杂的事情。

一、预测变化

        软件系统在发布后还可以不断地修改和演进,这就意味着不断有新的需求需要实现。预测变化的复杂性在于:

  • 不能每个设计点都考虑可扩展性。
  • 不能完全不考虑可扩展性。
  • 所有的预测都存在出错的可能性。
二、应对变化

        第一种应对变化的常见方案是将“变化”封装在一个“变化层”,将不变的部分封装在一个独立的“稳定层”。

  • 系统需要拆分出变化层和稳定层
  • 需要设计变化层和稳定层之间的接口

        第二种常见的应对变化的方案是提炼出一个“抽象层”和一个“实现层”。抽象层是稳定的,实现层可以根据具体业务需要定制开发,当加入新的功能时,只需要增加新的实现,无须修改抽象层,例如设计模式中的策略模式。后续的读书笔记中详细展开说明吧。

追求安全、低成本、规模

一、安全

        从技术的角度来讲,安全可以分为两类:一类是功能上的安全,一类是架构上的安全。

        功能安全其实就是“防小偷”,从实现的角度来看,功能安全更多地是和具体的编码相关。

        架构安全就是“防强盗”,传统的架构安全主要依靠防火墙,防火墙最基本的功能就是隔离网络,通过将网络划分成不同的区域,制定出不同区域之间的访问控制策略来控制不同信任程度区域间传送的数据流。例如DDos攻击防护。

二、低成本

        成本可以简单的分为:机器硬件的成本、第三方软件的成本、人员开发和维护成本等等;

三、规模

        软件规模带来复杂度的主要原因就是“量变引起质变”:

  • 功能越来越多,导致系统复杂度指数级上升
  • 数据越来越多,系统复杂度发生质变

读书小结

        这次读书笔记,我主要分享了软件架构的出现的历史背景,列举了两次软件危机,软件架构设计的目的是为了解决系统复杂度带来的问题,以及系统的复杂度的来源等,下一次《从零开始学架构》读书笔记,我将分享软件架构设计的三个原则、架构设计流程等的读书心得。

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

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

相关文章

蓝桥杯 历届真题 杨辉三角形【第十二届】【省赛】【C组】

资源限制 内存限制:256.0MB C/C时间限制:1.0s Java时间限制:3.0s Python时间限制:5.0s 思路: 由于我第一写没考虑到大数据的原因,直接判断导致只得了40分,下面是我的代码: #…

uniapp中uni.navigateTo传递变量

效果展示: 核心代码: uniapp中uni.navigateTo传递变量 methods: {changePages(item) {setDatas("maintenanceFunName", JSON.stringify(item)).then((res) > {uni.navigateTo({url: /pages/PMS/maintenance/maintenanceTypes/maintenanceT…

vscode教程

个人笔记(整理不易,有帮助点个赞) 笔记目录:学习笔记目录_pytest和unittest、airtest_weixin_42717928的博客-CSDN博客 个人随笔:工作总结随笔_8、以前工作中都接触过哪些类型的测试文档-CSDN博客 目录 一&#xff1a…

Mac安装Docker提示Another application changed your Desktop configuration解决方案

1. 问题描述 Mac安装Docker后,提示Another application changed your Desktop configuration,Re-apply configurations无效 2. 解决方案 在终端执行下述命令即可解决: sudo ln -sf /Applications/Docker.app/Contents/Resources/bin/docke…

neo4j图数据库下载安装配置

neo4j下载地址Index of /doc/neo4j/3.5.8/ 1.说明:jdk 1.8 版本对应的 neo4j 数据库版本 推荐安装3.X版本 2.配置系统环境变量 3.启动 neo4j.bat console 4.访问

系统监测工具-tcpdump的使用

一个简单的tcpdump抓包过程。主要抓包观察三次握手,四次挥手的数据包 有两个程序:客户端和服务器两个程序 服务器端的ip地址使用的是回环地址127.0.0.1 端口号使用的是6000 tcpdump -i 指定用哪个网卡等,dstip地址端口指定抓取目的地址…

[图像处理] MFC载入图片并绘制ROI矩形

上一篇: [图像处理] MFC载入图片并进行二值化处理和灰度处理及其效果显示 文章目录 前言完整代码重要代码效果 前言 上一篇实现了MFC通过Picture控件载入图片。 这一篇实现ROI功能的第一部分,在Picture控件中,通过鼠标拖拽画出一个矩形。 完…

前端:SVG绘制流程图

效果 代码 html代码 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><title>SVG流程图示例</title><style>/* CSS 样式 */</style><script src"js/index.js"></script…

浪潮信息边缘服务器NE5260G7焕新升级 算力系统更开放、更强大

随着边缘计算场景的增加&#xff0c;再加上用户需求的升级&#xff0c;边缘服务器市场亦迎来了新的挑战&#xff0c;紧跟需求&#xff0c;浪潮信息亦有序通过技术创新提升服务器性能。日前&#xff0c;浪潮信息边缘服务器NE5260G7焕新升级&#xff0c;完成第五代英特尔至强处理…

JVM 全景图

今天我重新复习了一下 jvm 的一些知识点。我以前觉得 jvm 的知识点很多很碎&#xff0c;而且记起来很困难&#xff0c;但是今天我重新复习了一下&#xff0c;对这些知识点进行了简单的梳理之后&#xff0c;产生了不一样的看法。虽然 jvm 的知识点很碎&#xff0c;但是如果你真的…

基于R语言、MaxEnt模型融合技术的物种分布模拟、参数优化方法、结果分析制图与论文写作

第一章、理论篇&#xff1a;以问题导入的方式&#xff0c;深入掌握原理基础 什么是MaxEnt模型&#xff1f; MaxEnt模型的原理是什么&#xff1f;有哪些用途&#xff1f; MaxEnt运行需要哪些输入文件&#xff1f;注意那些事项&#xff1f; 融合R语言的MaxEnt模型的优势&…

科技云报道:卷完参数卷应用,大模型落地有眉目了?

科技云报道原创。 国内大模型战场的比拼正在进入新的阶段。 随着产业界对模型落地的态度逐渐回归理性&#xff0c;企业客户的认知从原来的“觉得大模型什么都能做”的阶段&#xff0c;已经收敛到“大模型能够给自身业务带来什么价值上了”。 2023 年下半年&#xff0c;不少企…

[羊城杯 2020]Easyphp2 ---不会编程的崽

摆烂一周了&#xff0c;继续更&#xff01;&#xff01;题目还是简单哦。 提示明显要我们修改数据包&#xff0c;第一反应是修改referer。试了一下不太对。url很可能存在文件包含 使用伪协议读取一下源码吧。它过滤了base64关键字。尝试url编码绕过&#xff0c;这里可以使用二…

uniApp移动端安卓中使用webview打开pdf文件是下载而不是预览解决方案

关键 使用到 pdf.js 第一步&#xff1a; 下载pdf.js 文件到项目根目录 也就是这个文件 附下载地址&#xff1a;uni-app-pdf: 在uni-app中使用pdf.js实现在手机上打开pdf 也可通过其他方法下载 如npm 第二步&#xff1a; 拷贝hybrid文件到项目根目录 第三步&#xff1a;…

HarmonyOS实战开发-通过screenshot模块实现屏幕截图 。

介绍 本示例展示全屏截图和屏幕局部截图。 本示例通过screenshot模块实现屏幕截图 &#xff0c;通过window模块实现隐私窗口切换&#xff0c;通过display模块查询当前隐私窗口。 效果预览 使用说明&#xff1a; 点击右上角图标打开弹窗&#xff0c;选择截屏&#xff0c;展示…

stable-diffusion-webui安装教程

现在AI开始进入绘画领域,并且能自动根据文本来创建图片出来,这是一个划时代的进步。 这时候,我也不能落后,要紧跟上时代的步伐,那么也来学习一下stable-diffusion的使用,这样也算多一项对技术的认识,提高对AI的认知。 从网上看到很多stable-diffusion-webui的安装,其…

为什么苹果 Mac 电脑需要使用清理软件?

尽管 Apple Mac 电脑因其卓越的性能、简洁高效的 macOS 操作系统及独特的美学设计备受全球用户青睐&#xff0c;但任何电子设备在长期使用后都难以避免面临系统资源日渐累积的问题。其中一个重要维护需求在于&#xff0c;随着使用时间的增长&#xff0c;Mac电脑可能会由于系统垃…

element-ui 在Popover弹框中使用Select选择器,Vue3

bug描述&#xff1a; 当选择完select的时候,popover也会退出。 解决&#xff1a; popover组件的的关闭是当点击组件外的元素时会关闭&#xff0c;select虽然是写在组件内的&#xff0c;但是select有一个默认属性teleported“true” 会把它默认插到 body 元素&#xff0c;我…

Docker容器(五)Docker Compose

一、概述 1.1介绍 Docker Compose是Docker官方的开源项目&#xff0c;负责实现对Docker容器集群的快速编排。Compose 是 Docker 公司推出的一个工具软件&#xff0c;可以管理多个 Docker 容器组成一个应用。你需要定义一个 YAML 格式的配置文件docker-compose.yml&#xff0c;…

分布式主键ID生成策略

业务系统对分布式ID的要求 唯一性&#xff1a;在分布式系统中&#xff0c;每个节点都需要生成唯一的标识符来确保数据的唯一性。传统的单点生成ID方式无法满足分布式环境下的需求&#xff0c;而分布式ID能够在整个系统中保证每个节点生成的ID都是唯一的。 顺序性&#xff1a;某…