”工欲善其事,必先利其器。“—孔子《论语.录灵公》
首页 > 编程 > 与软件复杂性的永无休止的斗争

与软件复杂性的永无休止的斗争

发布于2024-08-18
浏览:205

The Never Ending Battle Against Software Complexity

什么是复杂性?

最近读完了《软件设计哲学》,第二章探讨了软件复杂性的话题。 

《软件设计哲学》一书实用地定义了复杂性:

“复杂性是与软件系统的结构相关的任何使其难以理解和修改的事物。”

换句话说,复杂性可以有多种形式,并且不一定与性能有任何关系,您的代码可以具有高性能但仍然很复杂

我想在本文中分享本书中的一些关键定义和见解。但首先,让我们想象一个您可能已经经历过的常见情况……


一个简短的恐怖故事

让我们深入探讨一个你们中的许多人可能经历过或将要经历的恐怖故事。

  1. 它从一个简单的 CRUD 任务管理应用程序开始。代码干净、模块化且易于维护。开发团队很高兴,系统对于最初的客户来说运行得很好。

  2. 当销售团队将系统出售给一家大公司时,问题就开始了,声称它具有日历集成、电子邮件通知和令人惊叹的报告生成器。销售完成后,必须快速实施这些功能。

  3. 日历集成: 团队必须与 Google 日历和 Outlook 集成。不同的开发人员实施了解决方案,导致方法不一致。

  4. 电子邮件通知: 接下来添加电子邮件通知。一位开发人员使用特定的库,而另一位开发人员创建了自定义解决方案。混合方法使代码变得混乱。

  5. 报告生成器: 对于报告生成器,开发人员使用了各种技术:PDF、Excel 导出和交互式仪表板。缺乏统一的方法使维护成为一场噩梦。

  6. 复杂性不断增加:每个功能都是独立快速开发的,导致功能之间存在依赖关系。开发人员开始创建“快速修复”以使一切正常运行,从而增加了系统的复杂性和耦合性。

软件开发不是在真空中进行的;而是在现实中进行的。各种内部和外部因素都会对其产生影响。我们都曾经历过或将会经历过这样的情况。


结束的开始

然后问题开始了:

  1. 系统某一部分的变化意外地影响了其他部分。
  2. 小改动需要修改许多其他文件,使得估计变得困难。
  3. 月复一月,代码变得越来越难以理解,通常通过反复试验来修复。
  4. 生产力下降,每个人都害怕维护任务。
  5. 不可避免地呼吁“我们需要重构。”
  6. 某些任务只能由特定的开发人员处理(经典)
  7. 随着时间的推移,曾经编写精美且文档齐全的软件变成了火车残骸。

命名症状

很明显,我们现在拥有一个复杂的系统。

现在让我们“剖析”这种复杂性,以便更容易识别和减轻它。

嗯,“缓解”的意思是:

“减轻严重性、严重性或痛苦;减轻。”

我相信复杂性通常是代码所固有的。有些事情本质上是复杂的。作为开发人员,您的角色不仅仅是创建计算机可以高效执行的代码,还要创建未来的开发人员(包括未来的您)可以使用的代码。

“控制复杂性是计算机编程的本质。”

— 布莱恩·科尼汉

上述书籍的作者指出,复杂性通常以三种方式表现出来,我们将在这里探讨。

改变放大

当看似简单的更改需要在许多不同的地方进行修改时,就会发生更改放大。

例如,如果产品负责人请求“优先级”或“完成日期”字段,并且您的实体紧密耦合,那么您需要进行多少更改?

认知负荷

认知负荷是指开发人员完成任务所需的知识量和时间。

想象一下这样的场景:一位新开发人员加入了团队,他被指派修复报告生成器中的错误。为了完成此任务,开发人员需要:

  • 了解不同的日历集成(Google 和 Outlook)。
  • 掌握电子邮件通知的不同方法。
  • 浏览报告生成器的碎片代码,处理 PDF、Excel 和仪表板。
  • 整合这些不同的技术和风格来查找并修复错误。

这是典型的“无法估计”场景,任务可能需要 1 分或 8 分——最好掷 D20 并做出相应的反应。

未知的未知

未知的未知是指你不知道自己不知道的事情。

这是复杂性的最糟糕表现,因为你可能会改变不应该改变的东西,导致一切崩溃。

示例:开发人员修改了电子邮件发送代码以添加新通知,但没有意识到这会影响依赖于该函数的报告生成器。这给客户带来了重大问题,体现了紧急复杂性的最坏形式。

复杂性的原因

看完恐怖故事和三个主要症状,让我们看看是什么导致了复杂性。

1. 依赖关系

依赖关系在软件中至关重要,无法完全消除。它们允许系统的不同部分相互作用并一起运行。然而,如果管理不当,依赖关系会显着增加复杂性。

定义:

当代码无法单独理解或修改时,就存在依赖关系,需要考虑或修改相关代码。

依赖项类型:

  • 直接:模块A直接依赖模块B。
  • 传递性:模块A依赖于模块B,模块B又依赖于模块C。
  • 循环:模块A、B、C以循环方式相互依赖。

2. 默默无闻

当重要信息不明显时,就会出现模糊性。这可能会使代码库难以理解,从而导致认知负荷增加和未知未知的风险。

定义:

当重要信息不明显时,就会出现模糊性。

晦涩难懂的例子:

  • 命名不当:名称不明确的变量和函数。
  • 隐藏的副作用:执行意外操作的方法。
  • 全局状态: 过度使用全局变量。
  • 深层继承:行为分布在类层次结构中的多个级别。

请记住:复杂性是渐进的

  • 复杂性很少是由单个“错误”或错误决策引起的。
  • 随着时间的推移,由于错误的决策和依赖性,复杂性“缓慢”地增加。

因为它是增量的,所以很容易想到,“就这一次,没关系。”但积累起来后,仅修复一两个依赖项不会产生太大影响。

“软件工程中的一切都是权衡。”
——我不记得作者了

结论

我可以写很多你可能已经在互联网上看到的关于如何避免复杂性的规则、策略和框架:SOLID、设计模式、YAGNI、KISS 等。

但是,您可以将它们全部统一为一个指导原则(如“务实的程序员”中提到的。): “我正在实现的内容容易更改吗?” 如果答案是否定的,那么您可能正在增加复杂性。

确保您的代码易于更改可以简化维护,减少开发人员的认知负担,并使系统更具适应性且不易出错。

谢谢你!

版本声明 本文转载于:https://dev.to/juniorklawa/the-never-ending-battle-against-software-complexity-46k1?1如有侵犯,请联系[email protected]删除
最新教程 更多>
  • Java中如何使用观察者模式实现自定义事件?
    Java中如何使用观察者模式实现自定义事件?
    在Java 中创建自定义事件的自定义事件在许多编程场景中都是无关紧要的,使组件能够基于特定的触发器相互通信。本文旨在解决以下内容:问题语句我们如何在Java中实现自定义事件以促进基于特定事件的对象之间的交互,定义了管理订阅者的类界面。以下代码片段演示了如何使用观察者模式创建自定义事件: args)...
    编程 发布于2025-06-10
  • Java中Lambda表达式为何需要“final”或“有效final”变量?
    Java中Lambda表达式为何需要“final”或“有效final”变量?
    Lambda Expressions Require "Final" or "Effectively Final" VariablesThe error message "Variable used in lambda expression shou...
    编程 发布于2025-06-10
  • 如何使用替换指令在GO MOD中解析模块路径差异?
    如何使用替换指令在GO MOD中解析模块路径差异?
    在使用GO MOD时,在GO MOD 中克服模块路径差异时,可能会遇到冲突,其中3个Party Package将另一个PAXPANCE带有导入式套件之间的另一个软件包,并在导入式套件之间导入另一个软件包。如回声消息所证明的那样: go.etcd.io/bbolt [&&&&&&&&&&&&&&&&...
    编程 发布于2025-06-10
  • 版本5.6.5之前,使用current_timestamp与时间戳列的current_timestamp与时间戳列有什么限制?
    版本5.6.5之前,使用current_timestamp与时间戳列的current_timestamp与时间戳列有什么限制?
    在时间戳列上使用current_timestamp或MySQL版本中的current_timestamp或在5.6.5 此限制源于遗留实现的关注,这些限制需要对当前的_timestamp功能进行特定的实现。 创建表`foo`( `Productid` int(10)unsigned not n...
    编程 发布于2025-06-10
  • 在UTF8 MySQL表中正确将Latin1字符转换为UTF8的方法
    在UTF8 MySQL表中正确将Latin1字符转换为UTF8的方法
    在UTF8表中将latin1字符转换为utf8 ,您遇到了一个问题,其中含义的字符(例如,“jáuòiñe”)在utf8 table tabled tablesset中被extect(例如,“致电。为了解决此问题,您正在尝试使用“ mb_convert_encoding”和“ iconv”转换受...
    编程 发布于2025-06-10
  • Python环境变量的访问与管理方法
    Python环境变量的访问与管理方法
    Accessing Environment Variables in PythonTo access environment variables in Python, utilize the os.environ object, which represents a mapping of envir...
    编程 发布于2025-06-10
  • 如何克服PHP的功能重新定义限制?
    如何克服PHP的功能重新定义限制?
    克服PHP的函数重新定义限制在PHP中,多次定义一个相同名称的函数是一个no-no。尝试这样做,如提供的代码段所示,将导致可怕的“不能重新列出”错误。 但是,PHP工具腰带中有一个隐藏的宝石:runkit扩展。它使您能够灵活地重新定义函数。 runkit_function_renction_re...
    编程 发布于2025-06-10
  • 在JavaScript中如何并发运行异步操作并正确处理错误?
    在JavaScript中如何并发运行异步操作并正确处理错误?
    同意操作execution 在执行asynchronous操作时,相关的代码段落会遇到一个问题,当执行asynchronous操作:此实现在启动下一个操作之前依次等待每个操作的完成。要启用并发执行,需要进行修改的方法。 第一个解决方案试图通过获得每个操作的承诺来解决此问题,然后单独等待它们: co...
    编程 发布于2025-06-10
  • 在细胞编辑后,如何维护自定义的JTable细胞渲染?
    在细胞编辑后,如何维护自定义的JTable细胞渲染?
    在JTable中维护jtable单元格渲染后,在JTable中,在JTable中实现自定义单元格渲染和编辑功能可以增强用户体验。但是,至关重要的是要确保即使在编辑操作后也保留所需的格式。在设置用于格式化“价格”列的“价格”列,用户遇到的数字格式丢失的“价格”列的“价格”之后,问题在设置自定义单元格...
    编程 发布于2025-06-10
  • 您可以使用CSS在Chrome和Firefox中染色控制台输出吗?
    您可以使用CSS在Chrome和Firefox中染色控制台输出吗?
    在javascript console 中显示颜色是可以使用chrome的控制台显示彩色文本,例如红色的redors,for for for for错误消息?回答是的,可以使用CSS将颜色添加到Chrome和Firefox中的控制台显示的消息(版本31或更高版本)中。要实现这一目标,请使用以下模...
    编程 发布于2025-06-10
  • 大批
    大批
    [2 数组是对象,因此它们在JS中也具有方法。 切片(开始):在新数组中提取部分数组,而无需突变原始数组。 令ARR = ['a','b','c','d','e']; // USECASE:提取直到索引作...
    编程 发布于2025-06-10
  • 图片在Chrome中为何仍有边框?`border: none;`无效解决方案
    图片在Chrome中为何仍有边框?`border: none;`无效解决方案
    在chrome 在使用Chrome and IE9中的图像时遇到的一个频繁的问题是围绕图像的持续薄薄边框,尽管指定了图像,尽管指定了;和“边境:无;”在CSS中。要解决此问题,请考虑以下方法: Chrome具有忽略“ border:none; none;”的已知错误,风格。要解决此问题,请使用以下...
    编程 发布于2025-06-10
  • eval()vs. ast.literal_eval():对于用户输入,哪个Python函数更安全?
    eval()vs. ast.literal_eval():对于用户输入,哪个Python函数更安全?
    称量()和ast.literal_eval()中的Python Security 在使用用户输入时,必须优先确保安全性。强大的Python功能Eval()通常是作为潜在解决方案而出现的,但担心其潜在风险。本文深入研究了eval()和ast.literal_eval()之间的差异,突出显示其安全性含义...
    编程 发布于2025-06-10
  • 如何使用Python理解有效地创建字典?
    如何使用Python理解有效地创建字典?
    在python中,词典综合提供了一种生成新词典的简洁方法。尽管它们与列表综合相似,但存在一些显着差异。与问题所暗示的不同,您无法为钥匙创建字典理解。您必须明确指定键和值。 For example:d = {n: n**2 for n in range(5)}This creates a dicti...
    编程 发布于2025-06-10
  • Go语言垃圾回收如何处理切片内存?
    Go语言垃圾回收如何处理切片内存?
    Garbage Collection in Go Slices: A Detailed AnalysisIn Go, a slice is a dynamic array that references an underlying array.使用切片时,了解垃圾收集行为至关重要,以避免潜在的内存泄...
    编程 发布于2025-06-10

免责声明: 提供的所有资源部分来自互联网,如果有侵犯您的版权或其他权益,请说明详细缘由并提供版权或权益证明然后发到邮箱:[email protected] 我们会第一时间内为您处理。

Copyright© 2022 湘ICP备2022001581号-3