优秀软件设计特征与原则
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格式笔记和代码自…...
解决Ubuntu22.04 VMware失败的问题 ubuntu入门之二十八
现象1 打开VMware失败 Ubuntu升级之后打开VMware上报需要安装vmmon和vmnet,点击确认后如下提示 最终上报fail 解决方法 内核升级导致,需要在新内核下重新下载编译安装 查看版本 $ vmware -v VMware Workstation 17.5.1 build-23298084$ lsb_release…...
STM32F4基本定时器使用和原理详解
STM32F4基本定时器使用和原理详解 前言如何确定定时器挂载在哪条时钟线上配置及使用方法参数配置PrescalerCounter ModeCounter Periodauto-reload preloadTrigger Event Selection 中断配置生成的代码及使用方法初始化代码基本定时器触发DCA或者ADC的代码讲解中断代码定时启动…...
对WWDC 2025 Keynote 内容的预测
借助我们以往对苹果公司发展路径的深入研究经验,以及大语言模型的分析能力,我们系统梳理了多年来苹果 WWDC 主题演讲的规律。在 WWDC 2025 即将揭幕之际,我们让 ChatGPT 对今年的 Keynote 内容进行了一个初步预测,聊作存档。等到明…...
HTML前端开发:JavaScript 常用事件详解
作为前端开发的核心,JavaScript 事件是用户与网页交互的基础。以下是常见事件的详细说明和用法示例: 1. onclick - 点击事件 当元素被单击时触发(左键点击) button.onclick function() {alert("按钮被点击了!&…...
今日科技热点速览
🔥 今日科技热点速览 🎮 任天堂Switch 2 正式发售 任天堂新一代游戏主机 Switch 2 今日正式上线发售,主打更强图形性能与沉浸式体验,支持多模态交互,受到全球玩家热捧 。 🤖 人工智能持续突破 DeepSeek-R1&…...
手机平板能效生态设计指令EU 2023/1670标准解读
手机平板能效生态设计指令EU 2023/1670标准解读 以下是针对欧盟《手机和平板电脑生态设计法规》(EU) 2023/1670 的核心解读,综合法规核心要求、最新修正及企业合规要点: 一、法规背景与目标 生效与强制时间 发布于2023年8月31日(OJ公报&…...
Python网页自动化Selenium中文文档
1. 安装 1.1. 安装 Selenium Python bindings 提供了一个简单的API,让你使用Selenium WebDriver来编写功能/校验测试。 通过Selenium Python的API,你可以非常直观的使用Selenium WebDriver的所有功能。 Selenium Python bindings 使用非常简洁方便的A…...
【若依】框架项目部署笔记
参考【SpringBoot】【Vue】项目部署_no main manifest attribute, in springboot-0.0.1-sn-CSDN博客 多一个redis安装 准备工作: 压缩包下载:http://download.redis.io/releases 1. 上传压缩包,并进入压缩包所在目录,解压到目标…...
GAN模式奔溃的探讨论文综述(一)
简介 简介:今天带来一篇关于GAN的,对于模式奔溃的一个探讨的一个问题,帮助大家更好的解决训练中遇到的一个难题。 论文题目:An in-depth review and analysis of mode collapse in GAN 期刊:Machine Learning 链接:...
从零手写Java版本的LSM Tree (一):LSM Tree 概述
🔥 推荐一个高质量的Java LSM Tree开源项目! https://github.com/brianxiadong/java-lsm-tree java-lsm-tree 是一个从零实现的Log-Structured Merge Tree,专为高并发写入场景设计。 核心亮点: ⚡ 极致性能:写入速度超…...
