优秀软件设计特征与原则
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个, 这几十个设计模式大致可以归为三大类: 创建型设计模式、结构型设计模式和行为型设计模式, 在后面的学习分享中会逐步展开。
相关文章:

优秀软件设计特征与原则
1.摘要 一款软件产品好不好用, 除了拥有丰富的功能和人性化的界面设计之外, 还有其深厚的底层基础, 而设计模式和算法是构建这个底层基础的基石。好的设计模式能够让产品开发快速迭代且稳定可靠, 迅速抢占市场先机;而好的算法能够让产品具有核心价值, 例如字节跳动…...

设备管理系统-设备管理软件
一、为什么要使用设备管理系统 1.企业扩张快,设备配置多,管理混乱。 2.设备数量多,存放地点多,查找麻烦。 3.同类设备单独管理, 困难。 4.设备较多时相关信息统计容易出错,错误后修改困难。 二、凡尔码设备管理软件的…...

物联网AI MicroPython学习之语法 I2S音频总线接口
学物联网,来万物简单IoT物联网!! I2S 介绍 模块功能: I2S音频总线驱动模块 接口说明 I2S - 构建I2S对象 函数原型:I2S(id, sck, ws, sd, mode, bits, format, rate, ibuf)参数说明: 参数类型必选参数?…...

Day31| Leetcode 455. 分发饼干 Leetcode 376. 摆动序列 Leetcode 53. 最大子数组和
进入贪心了,我觉得本专题是最烧脑的专题 Leetcode 455. 分发饼干 题目链接 455 分发饼干 让大的饼干去满足需求量大的孩子即是本题的思路: class Solution { public:int findContentChildren(vector<int>& g, vector<int>& s) {…...

基于C#实现赫夫曼树
赫夫曼树又称最优二叉树,也就是带权路径最短的树,对于赫夫曼树,我想大家对它是非常的熟悉,也知道它的应用场景,但是有没有自己亲手写过,这个我就不清楚了,不管以前写没写,这一篇我们…...
Android 13.0 app进程保活白名单功能实现
1.前言 在13.0的系统rom产品开发中,在某些重要的app即使进入后台,产品需求要求也不想被系统杀掉进程,需要app长时间保活,就是app进程保活白名单功能的实现, 所以需要在系统杀进程的时候不杀掉白名单的进程,接下来就看怎么样来实现这些功能 2.app进程保活白名单功能实…...
查找学习笔记
1、静态查找表 以下查找的索引均从1开始 (1)顺序查找(带哨兵) #include<iostream> #include<vector>using namespace std;int search(vector<int> arr, int key) {arr[0] key;int i;for (i arr.size() - 1…...
Qt QIODevice介绍
作者:令狐掌门 技术交流QQ群:675120140 csdn博客:https://mingshiqiang.blog.csdn.net/ 文章目录 主要功能用法示例读取数据写入数据使用数据流基于套接字的读写注意事项QIODevice 是 Qt 中所有输入/输出设备的抽象基类。它为派生类提供了一组标准的接口用于读写数据。这些派…...

python -opencv 中值滤波 ,均值滤波,高斯滤波实战
python -opencv 中值滤波 ,均值滤波,高斯滤波实战 cv2.blur-均值滤波 cv2.medianBlur-中值滤波 cv2.GaussianBlur-高斯滤波 直接看代码吧,代码很简单: import copy import math import matplotlib.pyplot as plt import matp…...

【教学类-06-07】20231124 (55格版)X-X之间的加法、减法、加减混合题
背景需求 在大四班里,预测试55格“5以内、10以内、20以内的加法题、减法题、加减混合题”的“实用性”。 由于只打印一份20以内加法减法混合题。 “这套20以内的加减法最难”,我询问谁会做(摸底幼儿的水平) 有两位男孩举手想挑…...
postgresql经常出现连接一会后服务器拒绝连接
本地连接远程Linux上PG数据库经常自动断开连接 原因:Linux设置的tcp的keepalive超时时间太长,如果网络状况不佳,可能会导致连接断掉。 [rootlocalhost ~]# sysctl -a | grep net.ipv4.tcp_keepalive sysctl: reading key "net.ipv6.con…...

迈巴赫S480升级主动式氛围灯 浪漫婉转的气氛
主动式氛围灯有263个可多色渐变的LED光源,营造出全情沉浸的动态光影氛围。结合智能驾驶辅助系统,可在转向或检测到危险时,予以红色环境光提示,令光影艺术彰显智能魅力。配件有6个氛围灯,1个电脑模块。 1、气候…...

Leetcode103 二叉树的锯齿形层序遍历
二叉树的锯齿形层序遍历 题解1 层序遍历双向队列 给你二叉树的根节点 root ,返回其节点值的 锯齿形层序遍历 。(即先从左往右,再从右往左进行下一层遍历,以此类推,层与层之间交替进行)。 提示:…...

可观测性建设实践之 - 日志分析的权衡取舍
指标、日志、链路是服务可观测性的三大支柱,在服务稳定性保障中,通常指标侧重于发现故障和问题,日志和链路分析侧重于定位和分析问题,其中日志实际上是串联这三大维度的一个良好桥梁。 但日志分析往往面临成本和效果之间的权衡问…...
Ceres使用
之前用过Ceres,但是只是跑例程,现在来着重学习一下使用流程。 1. 解决的问题 主要解决非线性优化问题。Ceres是一个较为通用的库。 参考链接 2. 如何使用 这个是求解的函数,主要关注这三个参数 CERES_EXPORT void Solve(const Solver::O…...

深度学习第1天:深度学习入门-Keras与典型神经网络结构
☁️主页 Nowl 🔥专栏《机器学习实战》 《机器学习》 📑君子坐而论道,少年起而行之 文章目录 神经网络 介绍 结构 基本要素 Keras 介绍 导入 定义网络 模型训练 前馈神经网络 特点 常见类型 代码示例 反馈神经网络 特点 …...

青云科技容器平台与星辰天合存储产品完成兼容性互认证
近日, 北京青云科技股份有限公司(以下简称:青云科技)的 KubeSphere 企业版容器平台成功完成了与 XSKY星辰天合的企业级分布式统一数据平台 V6(简称:XEDP)以及天合翔宇分布式存储系统 V6…...
谈谈基于Redis的分布式锁
目录 前言 基本介绍 演化过程 防死锁 防误删 自动续期 可重入 主从一致 总结 前言 在我们没有了解分布式锁前,使用最多的就是线程锁和进程锁,但他们仅能满足在单机jvm或者同一个操作系统下,才能有效。跨jvm系统,无法…...

逸学java【初级菜鸟篇】10.I/O(输入/输出)
hi,我是逸尘,一起学java吧 目标(任务驱动) 1.请重点的掌握I/O的。 场景:最近你在企业也想搞一个短视频又想搞一个存储的云盘,你一听回想到自己对于这些存储的基础还不是很清楚,于是回家开始了…...

【Python进阶笔记】md文档笔记第6篇:Python进程和多线程使用(图文和代码)
本文从14大模块展示了python高级用的应用。分别有Linux命令,多任务编程、网络编程、Http协议和静态Web编程、htmlcss、JavaScript、jQuery、MySql数据库的各种用法、python的闭包和装饰器、mini-web框架、正则表达式等相关文章的详细讲述。 全套md格式笔记和代码自…...

stm32G473的flash模式是单bank还是双bank?
今天突然有人stm32G473的flash模式是单bank还是双bank?由于时间太久,我真忘记了。搜搜发现,还真有人和我一样。见下面的链接:https://shequ.stmicroelectronics.cn/forum.php?modviewthread&tid644563 根据STM32G4系列参考手…...
SciencePlots——绘制论文中的图片
文章目录 安装一、风格二、1 资源 安装 # 安装最新版 pip install githttps://github.com/garrettj403/SciencePlots.git# 安装稳定版 pip install SciencePlots一、风格 简单好用的深度学习论文绘图专用工具包–Science Plot 二、 1 资源 论文绘图神器来了:一行…...

AI Agent与Agentic AI:原理、应用、挑战与未来展望
文章目录 一、引言二、AI Agent与Agentic AI的兴起2.1 技术契机与生态成熟2.2 Agent的定义与特征2.3 Agent的发展历程 三、AI Agent的核心技术栈解密3.1 感知模块代码示例:使用Python和OpenCV进行图像识别 3.2 认知与决策模块代码示例:使用OpenAI GPT-3进…...
连锁超市冷库节能解决方案:如何实现超市降本增效
在连锁超市冷库运营中,高能耗、设备损耗快、人工管理低效等问题长期困扰企业。御控冷库节能解决方案通过智能控制化霜、按需化霜、实时监控、故障诊断、自动预警、远程控制开关六大核心技术,实现年省电费15%-60%,且不改动原有装备、安装快捷、…...

【SQL学习笔记1】增删改查+多表连接全解析(内附SQL免费在线练习工具)
可以使用Sqliteviz这个网站免费编写sql语句,它能够让用户直接在浏览器内练习SQL的语法,不需要安装任何软件。 链接如下: sqliteviz 注意: 在转写SQL语法时,关键字之间有一个特定的顺序,这个顺序会影响到…...
使用van-uploader 的UI组件,结合vue2如何实现图片上传组件的封装
以下是基于 vant-ui(适配 Vue2 版本 )实现截图中照片上传预览、删除功能,并封装成可复用组件的完整代码,包含样式和逻辑实现,可直接在 Vue2 项目中使用: 1. 封装的图片上传组件 ImageUploader.vue <te…...
在Ubuntu中设置开机自动运行(sudo)指令的指南
在Ubuntu系统中,有时需要在系统启动时自动执行某些命令,特别是需要 sudo权限的指令。为了实现这一功能,可以使用多种方法,包括编写Systemd服务、配置 rc.local文件或使用 cron任务计划。本文将详细介绍这些方法,并提供…...

《基于Apache Flink的流处理》笔记
思维导图 1-3 章 4-7章 8-11 章 参考资料 源码: https://github.com/streaming-with-flink 博客 https://flink.apache.org/bloghttps://www.ververica.com/blog 聚会及会议 https://flink-forward.orghttps://www.meetup.com/topics/apache-flink https://n…...

OPenCV CUDA模块图像处理-----对图像执行 均值漂移滤波(Mean Shift Filtering)函数meanShiftFiltering()
操作系统:ubuntu22.04 OpenCV版本:OpenCV4.9 IDE:Visual Studio Code 编程语言:C11 算法描述 在 GPU 上对图像执行 均值漂移滤波(Mean Shift Filtering),用于图像分割或平滑处理。 该函数将输入图像中的…...
Hive 存储格式深度解析:从 TextFile 到 ORC,如何选对数据存储方案?
在大数据处理领域,Hive 作为 Hadoop 生态中重要的数据仓库工具,其存储格式的选择直接影响数据存储成本、查询效率和计算资源消耗。面对 TextFile、SequenceFile、Parquet、RCFile、ORC 等多种存储格式,很多开发者常常陷入选择困境。本文将从底…...