优秀软件设计特征与原则

1.摘要

一款软件产品好不好用, 除了拥有丰富的功能和人性化的界面设计之外, 还有其深厚的底层基础, 而设计模式和算法是构建这个底层基础的基石。好的设计模式能够让产品开发快速迭代且稳定可靠, 迅速抢占市场先机;而好的算法能够让产品具有核心价值, 例如字节跳动公司旗下的抖音、今日头条等众多产品以算法起家, 能够智能根据用户喜好精准推送其感兴趣的内容。在本章节中, 将对软件设计模式的相关知识进行总结, 为后面的学习打下基础。

2.设计模式简介

软件设计模式是指在软件开发过程中, 经过验证的, 用于解决在特定环境中重复出现的特定问题解决方案。可以将设计模式想象成根据需求进行调整的预制蓝图, 可用于解决代码中反复出现的设计问题。

设计模式与方法或库的使用方式不同, 很难直接在自己的程序中套用某个设计模式。模式并不是一段特定的代码, 而是解决特定问题的一般性概念。每一个程序员都可以根据模式来实现符合自己程序实际所需的解决方案。

2.1 为什么需要设计模式

优秀的开发者身上总蕴藏着一股力量, 阅读他们写的代码的最直观感受就是代码可重用性高、可读性强、灵活性好、可维护性强等特点。这些优秀的开发者实际上遵循的是他们自己的一套不断完善和总结的解决方案, 而设计模式实际上是根据以前的实践和经验记录要采用的解决方案, 在设计模式的实现过程中, 需要使用多个软件组件共同实现某些功能。因此, 设计模式加快了涉及多个组件的开发过程。开发者可以在对应解决方案的具体应用中使用自己熟悉的编程语言。

设计模式提供了经过验证的开发范例, 有助于节省时间, 而不必在每次出现问题时都重新创建设计模式。设计模式使程序设计更加标准化、代码编写更加工程化,从而提高软件的开发效率, 缩短软件的开发周期。

2.2 设计模式的发明者

在1994年, Erich Gamma、Richard Helm、Ralph Johnson和John Vlissides四人合著出版了<<设计模式-可复用面向对象软件的基础>>一书, 将设计模式的概念应用到程序开发领域中, 该书提供了23个模式来解决面向对象程序设计中的各种问题, 很快便成为了畅销书。由于书名太长, 人们将其简称为"四人组(Gang of Four, GoF)的书",并且很快进一步简化为''GoF的书"。

这些设计模式总共可以分为三大类: 创建型设计模式(Creational Patterns)、结构型设计模式(Structural Patterns)、行为型设计模式(Behavioral Patterns)。随着设计模式的不断发展, 设计模式的种类有所增加, 新增了空对象模式(Null Object Pattern)、规格模式(Specification)等。

2.3 怎样使用设计模式

在软件开发过程中, 开发者通常会基于业务需求选择设计模式, 在使用设计模式前, 开发者需要明白技术的目的是为业务而服务, 技术只是满足业务的一个工具, 如果开发者掌握了每种设计模式的应用场景、特征、优点和缺点, 以及不同设计模式的关联关系, 就可以很好的使用设计模式满足日常业务的需要。

2.3.1 需求驱动

在使用设计模式进行软件开发时, 应尽量按照特定需求进行综合分析和权衡。需求驱动应综合考虑软件的可维护性、可复用性等因素, 既要考虑开发效率, 又要考虑后期维护的便利性和复杂性。此外, 设计模式要根据具体项目进行评估, 如果某个项目没有应用场景, 则不一定需要设计模式。

2.3.2 对开发语言特性了解

设计模式在不同语言中的具体实现方式可能有所不同, 要根据具体的开发语言进行实现。例如:与Java、C++不同, Go语言中没有继承, 所以Go语言的设计模式具体实现方式与Java、C++使用的设计模式具体实现方式不同。

2.3.3 积累设计模式经验

学习编程的快速方法是进行实战, 学习设计模式也是如此, 在进行软件开发过程中多问问自己, 为什么要这样使用设计模式? 为什么要使用这个设计模式? 一定要使用这个设计模式吗?

2.3.4 避免设计过度

设计模式解决的是软件设计不科学问题, 但是在实战开发过程中, 容易出现设计过度问题。在设计模式的实战开发过程中, 核心原则是保持简洁, 设计模式的目的是使软件的设计及维护更加简单, 而不是更加复杂。

3.优秀设计特征

3.1 代码复用

无论是开发何种软件产品, 成本和时间都是重要的两个维度, 较短的开发时间意味着可比竞争对手更早进入市场, 抢占市场先机; 而较低的开发成本意味着能够留出更多营销资金, 因此能更广泛的覆盖潜在客户。

代码复用是减少开发成本时最常用的方式之一。其意图非常明显: 与其反复从头开发, 不如在新对象中重用已有代码。这个想法虽然可以, 但实际上要让已有代码在全新的上下文工作, 通常还是需要付出额外努力的。组件之间紧密的耦合、对具体类而非接口的依赖和硬编码的行为都会降低代码的灵活性, 使得复用这些代码变得更加困难。

使用设计模式是增加软件组件灵活性并使其易于复用的方式之一, 但有时, 这也会让组件变得更加复杂, 有时候不得不从底层、中间层和框架之间寻找平衡点, 因此一个好的代码复用解决方案总是从实践当中不断总结, 而不是生搬硬套设计模式。

3.2 扩展性

程序员生命中唯一不变的事情就是适应变化。

从程序员自身角度, 经常会遇到潜在变化的需求, 例如:

  • 在Windows平台上发布了一款应用, 但受众人群也想要macos平台的版本。

  • 我们创建了一个使用方形按钮的GUI框架, 但几个月之后圆形半透明按钮开始流行起来。

  • 设计了一款优秀的电子商务网站架构, 但仅仅几个月之后, 客户要求新增接受电话订单的功能。

相信每个开发者都在经历相似的事情, 为了适应变化, 我们会不断的优化代码结构, 评估现有的技术框架, 甚至给代码质量制定标准。最终目的是让自己的代码能够在不做大量更改的同时, 能够方便加入新的需求功能。

4.设计模式原则

在前人的基础上, 已经总结出一些公认的通用设计原则, 这里做个简单介绍, 后面的章节会详细展开。

4.1 单一职责原则

单一职责原则(Single Responsibility Principle)的主要目的是减少复杂度, 我们不需要费尽心机去构思如何使用200行代码来实现一个复杂设计, 实际上完全可以使用十几个清晰的方法。

当程序规模不断扩大、变更不断增加后, 真实问题才会逐渐显现出来, 到某个阶段, 相应的代码会变得过于庞大, 以至于无法记住所有细节,查找代码变得异常缓慢, 必须浏览整个类, 甚至整个工程才能找到需要的东西。如果开始感到对代码逐步失去控制, 应该回忆一下单一职责原则, 将某些类进行拆分, 尽量让一个类只做一件事情。

4.2 开闭原则

开闭原则(Open/Closed Principle)是指对扩展开放, 对修改关闭。在程序需要进行扩展时, 不能修改原有的代码, 实现一个热插拔效果, 从而使程序的扩展性更好,易于维护和升级。要达到这样的效果, 开发者需要使用接口和抽象类。

4.3 里氏替换原则

里氏替换原则(Liskov Substitution Principle)是面向对象设计的基本原则之一。里氏替换原则告诉我们, 任何基类可以出现的地方, 子类一定会出现。里氏替换原则是继承复用的基石, 只有当子类可以替换基类且软件组件的功能不受影响时, 基类才能真正被复用, 使子类能够在基类的基础上增加新的行为。里氏替换原则是对开闭原则的补充, 实现开闭原则的关键步骤是抽象化, 而基类与子类的继承关系是抽象化的具体实现, 所以里氏替换原则是对实现抽象化的具体步骤的规范。

4.4 依赖倒置原则

依赖倒置原则(Dependency Inversion Principle)是指在设计代码架构时, 高层模块不应该依赖底层模块, 二者都应该依赖于抽象, 抽象不应该依赖于细节, 细节应该依赖于抽象。

依赖倒置原则好处:

  • 减少类之间的耦合性,提高系统的稳定性。

  • 降低并行开发引起的风险。

  • 提高代码的可读性和可维护性。

4.5 接口隔离原则

接口隔离原则(Interface Segregation Principle)是指使用多个隔离接口比使用单个隔离接口要好, 它的另一个含义是降低类之间的耦合度, 由此可见, 设计模式就是从大型软件架构出发, 便于升级和维护的软件设计思想, 它强调减少依赖, 降低耦合度。

4.6 迪米特法则

迪米特法则(Law of Demeter)又叫最少知识原则, 也就是说, 一个对象应当对其它对象尽可能少的了解。迪米特法则的目的在于降低类之间的耦合性, 由于每个类尽量减少对其它类的依赖, 因此,很容易使得系统的功能模块功能独立,相互之间不存在(或很少有)依赖关系。

4.7 合成复用原则

合成复用原则(Composite/Aggregate Reuse Principle, CARP)是指尽量使用对象组合(has-a)/聚合(contanis-a),而不是继承关系达到软件复用的目的, 可以使系统更加灵活, 降低类与类之间的耦合度, 一个类的变化对其它类造成的影响相对较少。

5.总结

在本章节中, 我们学习了设计模式的概念和使用方式, 总结了设计模式的7大原则, 通过对设计模式原则的了解, 大致可以清楚在平时开发项目中需要注意的一些设计问题, 然而真正在项目应用的设计模式至少有23个, 这几十个设计模式大致可以归为三大类: 创建型设计模式、结构型设计模式和行为型设计模式, 在后面的学习分享中会逐步展开。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【Linux】Linux权限管理

目录 一、Linux中权限的概念 二、 Linux下的用户 2.1 用户的类型 2.2 用户创建、切换和删除 2.2.1 useradd或adduser命令创建用户 2.2.2 passwd命令设置用户密码 2.2.3 userdel命令删除用户 2.2.4 su命令切换用户身份等来管理和操作用户 2.3 注意事项 三、权限的管理…

STM32 寄存器配置笔记——USART配置中断接收乒乓缓存处理

一、概述 本文主要介绍如何配置USART接收中断&#xff0c;使用乒乓缓存的设计接收数据并将其回显在PC 串口工具上。以stm32f10为例&#xff0c;配置USART1 9600波特率。具体配置参考上一章节STM32 寄存器配置笔记——USART配置 打印。 乒乓缓存的设计应用场景&#xff1a;当后面…

EEG 脑电信号处理合集(2): 信号预处理

脑电信号在采集完以后&#xff0c;需要进行一系列的预处理操作&#xff0c;然后才能用于后续的科学研究和计算。预处理是脑电信号分析最基本且重要的一步。基于python环境MNE库。 1 使用带通滤波器&#xff0c;信号滤波&#xff0c;去噪&#xff0c;去工频干扰 data_path sam…

C#,《小白学程序》第十五课:随机数(Random)第二,统计学初步,数据统计的计算方法与代码

1 文本格式 /// <summary> /// 《小白学程序》第十五课&#xff1a;随机数&#xff08;Random&#xff09;第二&#xff0c;统计学初步&#xff0c;数据统计的计算方法与代码 /// 用随机数做简单的统计并用图形显示统计结果。 /// </summary> /// <param name&q…

【最新版】SolidWorks 2023 SP5.0 完整版安装包+安装教程

分享模式&#xff1a;免费/绿色&#xff0c;按教程安装 下载地址&#xff1a; https://pan.xunlei.com/s/VNL0-DD_ogcRFwy-xi0HUtlyA1?pwdfzqw# 提取码&#xff1a;fzqw SOLIDWORKS 2023新版本对电脑配置要求 更多详细说明请去官网查看。 安装使用方法&#xff1a; 一、卸…

雅可比矩阵(Jacobian Matrix)

假设给定一个从n维欧式空间到m维欧式空间的变换: 雅可比矩阵就是将一阶偏导数排列成一个m行、n列形式的矩阵&#xff0c;记作&#xff1a; 举一个例子&#xff1a; 雅可比矩阵等于&#xff1a;

ubuntu挂载硬盘方法

1.关闭服务器加上新硬盘 2.启动服务器&#xff0c;以root用户登录 3.查看硬盘信息 fdisk -l4.格式化分区 找到需要分区的目录,并记录分区的uuid&#xff0c;用于后面修改/etc/fstab永久挂载配置文件 mkfs.ext4 /dev/nvme0n1 mkfs.ext4 /dev/nvme1n1 Filesystem UUID: a1c…

数据资产确权的难点

数据是企业的重要资产之一&#xff0c;但是许多企业对于这项资产在管理上都面临着一些挑战&#xff0c;其中最关键就是数据确权的问题。接下来&#xff0c;将探讨数据资产确权的难点&#xff0c;并提出相应的解决方案&#xff0c;一起来看吧。 首先介绍一下数据资产入表的背景以…

一键填充字幕——Arctime pro

之前的博客中&#xff0c;我们聊到了PR这款专业的视频制作软件&#xff0c;但是pr有许多的功能需要搭配使用&#xff0c;相信不少小伙伴在剪辑视频时会发现一个致命的问题&#xff0c;就是字幕编写。伴随着人们对字幕需求的逐渐增加&#xff0c;这款软件便应运而生~ 相信应该有…

Altium Designer学习笔记13

0603电容封装的画法&#xff1a; 再画下三极管SOT-23的三极管的封装图&#xff1a; 画出三极管的封装图&#xff1a; 在画图的过程中&#xff0c;遇到了一个问题&#xff0c;画闭环线路的时候&#xff0c;就会被自动删除&#xff0c;查出是这个地方的配置需要进行修改。 那这个…

带submodule的git仓库自动化一键git push、git pull脚本

前言 很久没写博客了&#xff0c;今天难得闲下来写一次。 不知道大家在使用git的时候有没有遇到过这样的问题&#xff1a;发现git submodule特别好用&#xff0c;适合用于满足同时开发和部署的需求&#xff0c;并且结构清晰&#xff0c;方便我们对整个代码层次有一个大概的了…

SpringBoot——定制错误页面及原理

优质博文&#xff1a;IT-BLOG-CN 一、SpringBoot 默认的错误处理机制 【1】浏览器返回的默认错误页面如下&#xff1a; ☞ 浏览器发送请求的请求头信息如下&#xff1a; text/html会在后面的源码分析中说到。 【2】如果是其他客户端&#xff0c;默认则响应错误的 JSON字符串&…

[每周一更]-(第74期):Docker-compose 部署Jenkins容器-英文版及错误纠错

1、前文概要 通过物理机部署Jenkins前文已经讲过&#xff08;地址&#xff1a;[Jenkins] 物理机 安装 Jenkins&#xff09;&#xff0c;也已经公司内部平稳运行若干年&#xff0c;考虑到容器化的使用场景&#xff0c;部分项目都采用容器运行&#xff0c;开始考虑部署容器化的J…

hdlbits系列verilog解答(Exams/m2014 q4h)-44

文章目录 一、问题描述二、verilog源码三、仿真结果 一、问题描述 实现以下电路&#xff1a; 二、verilog源码 module top_module (input in,output out);assign out in;endmodule三、仿真结果 转载请注明出处&#xff01;

(2023码蹄杯)省赛(初赛)第二场真题(原题)(题解+AC代码)

题目1&#xff1a;MC0214捡麦子 码题集OJ-捡麦子 (matiji.net) 思路: 1.第n米在前n-1米的基础上多加一个n个麦子&#xff0c;那么直接从1开始枚举&#xff0c;累加答案即可 AC_Code:C #include<bits/stdc.h> using namespace std;int main( ) {int n; cin>>n;…

计算机组成原理-固态硬盘SSD

文章目录 总览机械硬盘vs固态硬盘固态硬盘的结构固态硬盘与机械硬盘相比的特点磨损均衡技术例题 总览 机械硬盘vs固态硬盘 固态硬盘采用闪存技术&#xff0c;是电可擦除ROM 下图右边黑色的块块就是一块一块的闪存芯片 固态硬盘的结构 块大小16KB~512KB 页大小512B~4KB 对固…

数据库的事务的基本特性,事务的隔离级别,事务隔离级别如何在java代码中使用,使用MySQL数据库演示不同隔离级别下的并发问题

文章目录 数据库的事务的基本特性事务的四大特性(ACID)4.1、原子性&#xff08;Atomicity&#xff09;4.2、一致性&#xff08;Consistency&#xff09;4.3、隔离性&#xff08;Isolation&#xff09;4.4、持久性&#xff08;Durability&#xff09; 事务的隔离级别5.1、事务不…

电源控制系统架构(PCSA)之系统控制处理器组件

目录 6.4 系统控制处理器 6.4.1 SCP组件 SCP处理器Core SCP处理器Core选择 SCP处理器核内存 系统计数器和通用计时器 看门狗 电压调节器控制 时钟控制 系统控制 信息接口 电源策略单元 传感器控制 外设访问 系统访问 6.4 系统控制处理器 系统控制处理器(SCP)是…

Ubuntu系统安装docker

1.检查是否安装老版本 检查卸载老版本docker ubuntu下自带了docker的库&#xff0c;不需要添加新的源。 但是ubuntu自带的docker版本太低&#xff0c;需要先卸载旧的再安装新的。 apt-get remove docker docker-engine docker.io containerd runc 如果不能正常卸载&#x…

VINS-MONO代码解读----vins_estimator(鲁棒初始化部分)

0. 前言 整个初始化部分的pipeline如下所示&#xff0c;参照之前的博客&#xff0c;接下来根据代码一步步讲解。 1. 旋转约束标定旋转外参Rbc 上回讲了processImage中addFeatureCheckParallax完成了对KF的筛选&#xff0c;我们知道了2nd是否为KF&#xff0c;接下来是初始化…