《软件工程》第 13 章 - 软件维护
知识思维导图
13.1 软件维护与进化的概念
1. 核心概念
- 软件维护:软件交付使用后,为纠正错误、改善性能或其他属性而进行的修改过程
- 软件进化:随着时间推移,软件系统为适应环境变化和用户需求而不断演变的过程
2. 维护类型(Java 代码示例)
/*** 维护类型演示类*/
public class MaintenanceDemo {// 纠错性维护:修复除法溢出错误public double calculate(int a, int b) {if (b == 0) { // 纠错性维护新增的防御性代码throw new IllegalArgumentException("除数不能为0");}return a / b;}// 适应性维护:适配新操作系统的文件路径格式public String getFilePath(String fileName) {String os = System.getProperty("os.name").toLowerCase();if (os.contains("win")) { // 适应性维护的环境判断return "C:\\" + fileName;} else {return "/usr/local/" + fileName;}}// 完善性维护:新增日志功能public void processData(String data) {System.out.println("开始处理数据:" + data); // 完善性维护新增的日志// 原有处理逻辑}// 预防性维护:提前优化算法复杂度public List<Integer> optimizeAlgorithm(List<Integer> list) {// 预防性维护:将O(n²)算法优化为O(n log n)Collections.sort(list);return list;}
}
13.2 软件维护的过程模型
13.2.1 结构化与非结构化维护对比
维度 | 结构化维护 | 非结构化维护 |
---|---|---|
文档支持 | 完整的需求 / 设计 / 测试文档 | 缺乏或无文档 |
修改流程 | 遵循需求分析→设计→编码→测试流程 | 直接修改代码,无规范流程 |
维护成本 | 低 | 高 |
风险 | 可预测 | 不可预测,易引发副作用 |
13.2.2 维护成本公式
软件维护成本占比示意图
13.2.3 常见问题及解决方案
-
问题 1:人员流动导致知识断层
解决方案:建立维护知识库,录制代码讲解视频 -
问题 2:技术债累积
解决方案:制定技术债偿还计划,定期进行代码重构
13.3 可维护性
1. 可维护性三大指标
可维护性指标架构图
2. 提升可维护性的 Java 实践
// 可理解性:使用有意义的变量名和注释
public class UserService {// 可修改性:使用接口隔离原则private final UserRepository userRepository; // 依赖抽象而非具体实现public UserService(UserRepository userRepository) {this.userRepository = userRepository;}// 可测试性:分离业务逻辑与外部依赖public boolean validateUser(String userId) {if (StringUtils.isBlank(userId)) { // 使用工具类增强可读性log.error("用户ID为空");return false;}return userRepository.existsById(userId);}
}
13.4 维护活动及实施策略
1. 维护活动流程图
软件维护活动流程图
2. 版本控制最佳实践
# Git分支管理策略示例
git checkout -b feature/optimize-login # 新建功能优化分支
# 修改代码并测试通过
git commit -m "优化登录功能可维护性"
git checkout main
git merge feature/optimize-login # 合并到主分支
13.5 维护副作用
1. 副作用类型及预防措施
类型 | 示例 | 预防措施 |
---|---|---|
代码副作用 | 修改模块 A 导致模块 B 功能异常 | 编写单元测试,使用依赖注入 |
数据副作用 | 字段类型修改导致旧数据丢失 | 数据迁移前做备份,增加兼容性校验 |
文档副作用 | 代码修改后未更新设计文档 | 建立文档更新 checklist |
2. 副作用检测代码示例(JUnit 测试)
public class SideEffectTest {private MaintenanceDemo demo;@BeforeEachpublic void setUp() {demo = new MaintenanceDemo();}// 检测修改计算方法是否影响原有功能@Testpublic void testCalculateSideEffect() {assertDoesNotThrow(() -> demo.calculate(10, 2)); // 验证无异常抛出assertEquals(5.0, demo.calculate(10, 2), 0.01); // 验证结果正确性}
}
13.6 逆向工程与软件重构
1. 逆向工程步骤(以 Java 为例)
- 使用 JD-GUI 反编译.class 文件获取源代码
- 通过 UML 工具生成类图(示意图位置:图 4)
- 分析代码结构,推导设计模式
2. 代码重构实战:提取重复代码
重构前代码:
// 重复代码示例
public void saveUser(User user) {String sql = "INSERT INTO user VALUES(?, ?, ?)";// 执行数据库操作
}public void updateUser(User user) {String sql = "UPDATE user SET name=?, age=? WHERE id=?";// 执行数据库操作(与save方法重复)
}
重构后代码:
public class DatabaseHelper {private void executeSql(String sql, Object... params) {// 统一数据库操作逻辑}public void saveUser(User user) {String sql = "INSERT INTO user VALUES(?, ?, ?)";executeSql(sql, user.getId(), user.getName(), user.getAge());}public void updateUser(User user) {String sql = "UPDATE user SET name=?, age=? WHERE id=?";executeSql(sql, user.getName(), user.getAge(), user.getId());}
}
13.7总结
软件维护是软件生命周期中耗时最长、成本最高的阶段,通过结构化维护流程、提升代码可维护性、规范版本管理等策略,能够有效降低维护成本并减少副作用。逆向工程和重构技术则为老旧系统的现代化改造提供了重要手段。建议在实际项目中建立维护日志,定期进行维护复盘,持续优化软件系统的可维护性。
相关文章:

《软件工程》第 13 章 - 软件维护
知识思维导图 13.1 软件维护与进化的概念 1. 核心概念 软件维护:软件交付使用后,为纠正错误、改善性能或其他属性而进行的修改过程软件进化:随着时间推移,软件系统为适应环境变化和用户需求而不断演变的过程 2. 维护类型&#…...

2024 CKA模拟系统制作 | Step-By-Step | 12、创建多容器Pod
目录 免费获取题库配套 CKA_v1.31_模拟系统 一、题目 二、考点分析 1. 多容器 Pod 的理解 2. YAML 配置规范 3. 镜像版本控制 三、考点详细讲解 1. 多容器 Pod 的工作原理 2. 容器端口冲突处理 3. 资源隔离机制 四、实验环境搭建步骤 总结 免费获取题库配套 CKA_v…...

python:selenium爬取网站信息
关注我,精彩不错过! 前言 使用python的requests模块还是存在很大的局限性,例如:只发一次请求;针对ajax动态加载的网页则无法获取数据等等问题。特此,本章节将通过selenium模拟浏览器来完成更高级的爬虫抓…...
Nginx版本平滑迁移方案
Nginx版本平滑迁移方案 最可靠方案:make install后,先-s stop再重启,100%确保版本切换特殊情况:当发现nginx.pid.oldbin文件时,才考虑使用USR2信号无损升级避坑重点:make install只是替换文件,…...
WPF 按钮悬停动画效果实现
WPF 按钮悬停动画效果实现 下面我将实现一个专业的按钮悬停动画效果:当鼠标悬停在按钮上时,按钮上的文字由黑色变为白色,同时加粗并变大。 完整实现方案 MainWindow.xaml <Window x:Class"ButtonHoverEffect.MainWindow"xml…...

满天星之canvas实现【canvas】
展示 文章目录 展示Canvas 介绍【基础】简介兼容性关键特性注意事项应用场景:基本示例 满天星代码实现【重点】代码解释 全量代码【来吧,尽情复制吧少年】html引入JS代码 参考资源 Canvas 介绍【基础】 简介 Canvas是一个基于HTML5的绘图技术࿰…...
我在架构师面前谈 Spring Inner Beans,他直接点头说:这人有料!
“你听说了吗?阿里、字节最近的Java面试题又加难了!” “嗯?咋了?” “Spring又被拿出来问了,这次居然问到了Inner Beans!” “这不是冷门题吗?” “是啊,我一开始还真没答上来……” 是的!今天要跟大家唠嗑的,就是这个在面试中悄悄冒头,但平时开发中却经常被我们忽…...
Java无序数组 vs 有序数组:性能对比与选型指南
在Java中选择使用无序数组还是有序数组,需根据具体的应用需求和操作特性进行权衡。以下是从不同维度分析的详细对比及建议: 一、核心操作的性能对比 操作无序数组有序数组插入/追加O(1)(直接尾部插入)O(n)(需移动元素…...
【Linux 基础知识系列】第二篇-Linux 发行版概述
一、什么是 Linux 发行版? Linux 发行版是指将 Linux 内核和应用程序、工具、库等有机组合在一起,形成一个完整的操作系统。由于 Linux 的开源特性,任何人都可以在 Linux 内核的基础上进行修改和定制,因此产生了许多不同的发行版…...

【开源解析】基于PyQt5+Folium的谷歌地图应用开发:从入门到实战
🌐【开源解析】基于PyQt5Folium的谷歌地图应用开发:从入门到实战 🌈 个人主页:创客白泽 - CSDN博客 🔥 系列专栏:🐍《Python开源项目实战》 💡 热爱不止于代码,热情源自每…...

在 Ubuntu 22.04 LTS 上离线安装 Docker
在 Ubuntu 22.04 LTS 上离线安装 Docker 一、准备工作 1.1 获取目标系统信息 在目标 Ubuntu 22.04 LTS 系统上,先执行以下命令确认架构信息: uname -m lsb_release -a一般返回如下信息: 1.2 需要一台可联网的机器 准备一台可以连接互联网…...

python调用langchain实现RAG
一、安装langchain 安装依赖 python -m venv env.\env\Scripts\activatepip3 install langchainpip3 install langchain-corepip3 install langchain-openaipip3 install langchain-communitypip3 install dashscopepip3 install langchain_postgrespip3 install "psyc…...
Qt 中的 d-pointer 与 p-pointer小结
Qt 中的 d-pointer 与 p-pointer: PIMPL 惯用法解析 在 Qt 库中,尤其是在其核心类和模块中,广泛使用了 PIMPL (Pointer to IMPLementation,指向实现的指针) 的编程惯用法。这种模式通过一对指针来实现:d-pointer (d_p…...
冷库耗电高的一种重要原因分析,以及一种降低冷库电费≥20%的方法
1.冷库耗电高的原因分析 1.1化霜不及时 固定周期化霜:传统定时化霜模式(如每日2次)未考虑实际结霜量,导致无效化霜(霜层薄时仍启动化霜),浪费大量电能。 化霜时间过长:化霜加热器…...
理解 Redis 事务-21(使用事务实现原子操)
使用事务实现原子操作 Redis 事务是一种在单个步骤中执行一组命令的机制。"要么全部,要么全部不"的方法确保了数据的一致性和完整性,尤其是在需要对相关数据进行多个操作时。没有事务,并发操作可能会导致竞争条件和不一致的数据状…...
神经网络加上注意力机制,精度反而下降,为什么会这样呢?注意力机制的本质是什么?如何正确使用注意力机制?注意力机制 | 深度学习
在深度学习的发展中,注意力机制的引入曾被誉为一次划时代的技术飞跃。无论是在自然语言处理领域产生Transformer架构,还是在图像识别、语音识别和推荐系统等多个方向取得显著成效,注意力机制的价值似乎毋庸置疑。然而,在一些实际应用场景中,研究人员和工程师却发现:在传统…...

触控精灵 ADB运行模式填写电脑端IP教程
•ADB模式,如果你手机已经root则可以直接运行,无需安装电脑端。 •ADB模式,如果你手机没有root,那你可以windows电脑下载【极限投屏】软件,然后你的手机和电脑的网络要同一个wifi,然后把你电脑的ip地址填写…...

uniapp|实现多端图片上传、拍照上传自定义插入水印内容及拖拽自定义水印位置,实现水印相机、图片下载保存等功能
本文以基础视角,详细讲解如何在uni-app中实现图片上传→水印动态编辑→图片下载的全流程功能。 目录 引言应用场景分析(社交媒体、内容保护、企业素材管理等)uniapp跨平台开发优势核心功能实现图片上传模块多来源支持:相册选择(`uni.chooseImage`)与拍照(`sourceType:…...

linux有效裁剪视频的方式(基于ffmpeg,不改变分辨率,帧率,视频质量,不需要三方软件)
就是在Linux上使用OBS Studio录制一个讲座或者其他视频,可能总有些时候会多录制一段时间,但是如果使用剪映或者PR这样的工具在导出的时候总需要烦恼导出的格式和参数,比如剪映就不支持mkv格式的导出,导出成mp4格式的视频就会变得很…...

服务器密码安全运维解决新思路:凭据管理SMS+双因素SLA认证结合的方案
引言:云服务器安全成本困局 在云计算渗透率突破60%的今天,中小企业正面临严峻的安全悖论:某权威机构数据显示,72%的云上数据泄露事件源于凭据管理不当,而传统安全解决方案的采购成本往往超过中小企业年利润的8%。这种…...

论文阅读笔记——In-Context Edit
ICEdit 论文阅读笔记 指令图像编辑现有方法的局限: 微调类方法(InstructPix2Pix、Emu Edit、 Ultra Edit):需要大规模数据和算力、精度高但效率低且泛化性低;免训练方法(Prompt-to-Prompt、 StableFlow&am…...
Debian 系统 Python 开发全解析:从环境搭建到项目实战
Debian 系统 Python 开发全解析:从环境搭建到项目实战 在当今数字化时代,Python 凭借其简洁易读的语法和强大的功能,成为了最受欢迎的编程语言之一。Debian 作为一款稳定、安全且开源的 Linux 操作系统,为 Python 开发提供了理想的环境。本文将详细介绍在 Debian 系统上进…...
Next.js 15 与 Apollo Client 的现代集成及性能优化
Next.js 15 与 Apollo Client 的现代集成及性能优化 目录 技术演进集成实践性能优化应用案例未来趋势 技术演进 Next.js 15 核心特性对开发模式的革新 Next.js 15 通过引入 App Router、服务器组件(Server Components)和客户端组件(Clie…...

【后端高阶面经:MongoDB篇】41、MongoDB 是怎么做到高可用的?
一、MongoDB高可用核心架构:副本集(Replica Set)设计 (一)副本集角色与拓扑结构 1. 三大核心角色 角色职责描述资源占用选举权重数据存储Primary唯一接收写请求的节点,将操作日志(Oplog&…...
IO Vs NIO
一、IO(传统阻塞式) 全称:Input/Output(输入/输出) 定义:Java 1.0 引入的基础 I/O 模型,基于流(Stream)的同步阻塞操作,线程在读写数据时会阻塞直到操作完成。 二、NIO(新式非阻塞式) 全…...
offset 家族和 client 家族
在前端开发中,offset 家族和 client 家族是用于获取元素尺寸和位置的重要属性集合。以下是对这两个家族相关知识点的系统总结: 一、offset 家族 核心属性 offsetWidth / offsetHeight 含义:元素的总尺寸,包含内容区、内边距、边…...

DMBOK对比知识点整理(4)
1.常见数据质量维度 常见数据质量维度(DMBOK-P353)质量维度...

day12 leetcode-hot100-21(矩阵4)
240. 搜索二维矩阵 II - 力扣(LeetCode) 1.暴力法O(m*n) 思路:两层for循环即可。 2.二分查找O(m*logn) 思路:每行都用二分查找,因为每行都是排好序的 class Solution {public boolean searchMatrix(int[][] matrix, int targe…...
Java基础 Day24
一、进程和线程 1、进程 (1)概念 进程 (Process) 是计算机中的程序关于某数据集合上的一次运行活动 是系统进行资源分配的基本单位 简单理解:程序的执行过程(正在运行的应用程序) (2)特性…...

提问:鲜羊奶是解决育儿Bug的补丁吗?
在育儿这个"系统工程"中,过度提醒就像冗余代码:"快写作业"(重复调用)、"多穿衣服"(异常捕获)、"别玩手机"(进程阻断)。羊大师技术育儿实验…...