」工欲善其事,必先利其器。「—孔子《論語.錄靈公》
首頁 > 程式設計 > 大泥球:理解反模式以及如何避免它

大泥球:理解反模式以及如何避免它

發佈於2024-11-08
瀏覽:580

前端开发中最臭名昭著的架构反模式可能是大泥球。术语“大泥球”适用于没有明显结构或模块化组织的系统。代码库有机且混乱地增长,成为维护的噩梦。这是许多开发人员发现自己所处的情况,特别是当他们面临着按时完成任务并开发大量功能的压力时。
这就是当前文章的内容:大泥球反模式以及前端开发中的示例,为什么它如此常见,它何时成为问题以及如何解决这个问题。

Big Ball of Mud: Understanding the Antipattern and How to Avoid It

什么是大泥球?

大泥球是一个架构边界定义不明确的系统。在这样的系统中,代码变得纠缠和高度耦合,因此维护或扩展项目变得有问题。随着时间的推移,随着越来越多的功能被添加而不关注整体设计,使用代码变得越来越困难。如果没有结构,对系统某一部分的更改很容易破坏其他部分,从而无意中引入错误,从而进一步提高开发的复杂性。

大泥球中,你经常会看到以下特征:
NOAA 明确关注点分离;业务逻辑、UI 和数据获取是交织在一起的。 NOAA 松散耦合;这些组件相互交织,因此很难将更改分开。 NOAA 模块化;系统的每个部分都依赖于其他部分。 NOAA 全局变量或具有不可预测副作用的共享状态。

大泥球是快速交付而没有对架构给予应有关注的高压的常见结果。在项目开始时,开发人员通常急于尽快构建功能,而很少有时间进行充分的规划。这导致代码库在各个方向上增长,新逻辑被插入到任何适合的地方。随着时间的推移,重构会被延迟或忽略,以支持交付更多功能,并且架构会恶化。

造成这种反模式的其他因素包括:

  • 缺乏协调:开发者之间没有协调。不一致的编码和分散的功能随之而来。
  • 没有制定标准或架构原则来指导开发。
  • 技术债务:在已经很混乱的代码之上添加新功能,而没有清理混乱。

让我们仔细看看大泥球在普通前端项目中可能是什么样子。

前端大泥球示例

这是前端架构中大泥球反模式的抽象示例。考虑一个小型 React 项目,该项目在一段时间后变得混乱。

文件结构:

/src
  /components
    /Header.js
    /Footer.js
/Sidebar.js
    /MainContent.js
    /UserProfile.js
  /utils
    /api.js
    /constants.js
  /styles
    /globalStyles.css
  /App.js
  /index.js

该架构的问题:

  • 缺乏模块边界:组件,例如页眉、页脚和用户配置文件,都驻留在一个文件夹中,而不考虑它们各自所扮演的角色。
  • 关注点混合: 组件负责获取数据,即 API 调用和渲染 UI 元素。因此,逻辑层和表示层之间的紧密耦合仍然存在。
  • 全局样式: 该项目依赖于单个全局 CSS 文件。随着应用程序的增长,这可能会导致样式冲突,并且更难以维护。 在组件中直接使用API​​:在UserProfile.js等组件中直接导入获取和更新数据的方法,从而将数据获取逻辑与UI代码混合在一起。

UserProfile.js 的示例代码:

import React, { useState, useEffect } from 'react';
import { fetchUserData, updateUserProfile } from './utils/api';
import './styles/globalStyles.css'; 

const UserProfile = () => {
  const [user, setUser] = useState(null);
  const [loading, setLoading] = useState(true);

  useEffect(() => {
    fetchUserData()
    .then((data) => {
        setUser(data);
        setLoading(false);
      })
      .catch((error) => console.error('Error fetching user data:', error));
  }, []);

  const handleUpdate = () => {
    updateUserProfile(user)
      .then(() => alert('Profile updated!'))
      .catch((error) => console.error('Error updating profile:', error));
  };

  if (loading) return 
Loading.
; return (

{user.name}

); }; export default UserProfile;

代码中的问题:

  • 无 SoC: 数据获取、状态管理和 UI 渲染均在组件内的一处执行。
  • 紧耦合:API的更新会强制更新多个组件,因为API和组件之间没有抽象层。
  • 无逻辑重用:另一个需要访问用户数据的组件将重新实现 API 调用或将自身与此逻辑紧密耦合。

这种错综复杂、相互依赖的代码很难扩展和维护,这就是大泥球。

问题什么时候开始出现?

具有此类架构的项目可能不会立即出现明显的问题迹象。但随着项目的发展,问题也随之复杂化:

  • 开发速度缓慢:更改的风险更大,因为与更改位置无关的系统部分可能会出现错误。
  • 更大的技术债务:无需重构的附加功能集涉及架构改进,而这些改进变得更难以实现。
  • 生产力低下:开发人员将开始花费更多时间来导航并从这些混乱的代码中找出有意义的东西,从而减慢功能的开发。

打结越多,就越难解开。当然,这只是技术债务增加和生产力下降的恶性循环。

如何避免大泥球

为了避免“大泥球”,必须尽早灌输良好的架构习惯,并在开发过程中严格执行。以下是一些策略。

  1. 模块化架构:将代码清晰地划分为具有责任边界的逻辑模块。例如,关注点可以通过数据获取、业务逻辑和 UI 渲染来分离。

  2. 抽象:通过服务或挂钩抽象 API 调用和数据管理,以便将这些问题从组件中抽象出来。这将有助于解耦代码并更轻松地处理 API 中的更改。

  3. 模块边界:组件之间应该有明确定义的边界。不要将所有组件都放在一个文件夹下,而是为一项功能或域创建单独的文件夹。

  4. 全局状态管理: 使用 Redux、MobX 或 React 的 Context API 等库进行组件之间的共享状态管理。这大大减少了组件自行管理状态的需求。

  5. 重构:定期重构。不要让项目发展到绝对无法再处理的阶段;解决这些问题,同时保持代码库干净。

如果您已经陷入大泥球该怎么办

如果你的项目已经变成了一个大泥球,那么还有希望。补救措施是逐步重构代码库,在可能的情况下烘焙架构原则。开始于:

  • 识别痛点:重点关注代码中使用或扩展最困难的部分。
  • 模块化组件: 增量重构组件以分离关注点并限制依赖关系。 现在,介绍测试:添加单元和集成测试,以确保您的重构不会破坏现有功能。

总之,大泥球是一种非常常见的反模式,在前端项目中造成很多麻烦。模块化架构的引入、关注点分离和定期重构绝对是避免代码库中这种模式带来的混乱的步骤,使其更干净、更易于管理。

版本聲明 本文轉載於:https://dev.to/m_midas/big-ball-of-mud-understanding-the-antipattern-and-how-to-avoid-it-2i?1如有侵犯,請洽study_golang@163 .com刪除
最新教學 更多>
  • 在JavaScript中如何並發運行異步操作並正確處理錯誤?
    在JavaScript中如何並發運行異步操作並正確處理錯誤?
    同意操作execution 在執行asynchronous操作時,相關的代碼段落會遇到一個問題,當執行asynchronous操作:此實現在啟動下一個操作之前依次等待每個操作的完成。要啟用並發執行,需要進行修改的方法。 第一個解決方案試圖通過獲得每個操作的承諾來解決此問題,然後單獨等待它們: c...
    程式設計 發佈於2025-05-22
  • 如何從PHP中的Unicode字符串中有效地產生對URL友好的sl。
    如何從PHP中的Unicode字符串中有效地產生對URL友好的sl。
    為有效的slug生成首先,該函數用指定的分隔符替換所有非字母或數字字符。此步驟可確保slug遵守URL慣例。隨後,它採用ICONV函數將文本簡化為us-ascii兼容格式,從而允許更廣泛的字符集合兼容性。 接下來,該函數使用正則表達式刪除了不需要的字符,例如特殊字符和空格。此步驟可確保slug僅包...
    程式設計 發佈於2025-05-22
  • 在Java中使用for-to-loop和迭代器進行收集遍歷之間是否存在性能差異?
    在Java中使用for-to-loop和迭代器進行收集遍歷之間是否存在性能差異?
    For Each Loop vs. Iterator: Efficiency in Collection TraversalIntroductionWhen traversing a collection in Java, the choice arises between using a for-...
    程式設計 發佈於2025-05-22
  • 左連接為何在右表WHERE子句過濾時像內連接?
    左連接為何在右表WHERE子句過濾時像內連接?
    左JOIN CONUNDRUM:WITCHING小時在數據庫Wizard的領域中變成內在的加入很有趣,當將c.foobar條件放置在上面的Where子句中時,據說左聯接似乎會轉換為內部連接。僅當滿足A.Foo和C.Foobar標準時,才會返回結果。 為什麼要變形?關鍵在於其中的子句。當左聯接的右側...
    程式設計 發佈於2025-05-22
  • C++中如何將獨占指針作為函數或構造函數參數傳遞?
    C++中如何將獨占指針作為函數或構造函數參數傳遞?
    在構造函數和函數中將唯一的指數管理為參數 unique pointers( unique_ptr [2啟示。通過值: base(std :: simelor_ptr n) :next(std :: move(n)){} 此方法將唯一指針的所有權轉移到函數/對象。指針的內容被移至功能中,在操作...
    程式設計 發佈於2025-05-22
  • 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-05-22
  • 用戶本地時間格式及時區偏移顯示指南
    用戶本地時間格式及時區偏移顯示指南
    在用戶的語言環境格式中顯示日期/時間,並使用時間偏移在向最終用戶展示日期和時間時,以其localzone and格式顯示它們至關重要。這確保了不同地理位置的清晰度和無縫用戶體驗。以下是使用JavaScript實現此目的的方法。 方法:推薦方法是處理客戶端的Javascript中的日期/時間格式化和...
    程式設計 發佈於2025-05-22
  • 如何修復\“常規錯誤:2006 MySQL Server在插入數據時已經消失\”?
    如何修復\“常規錯誤:2006 MySQL Server在插入數據時已經消失\”?
    How to Resolve "General error: 2006 MySQL server has gone away" While Inserting RecordsIntroduction:Inserting data into a MySQL database can...
    程式設計 發佈於2025-05-22
  • Python讀取CSV文件UnicodeDecodeError終極解決方法
    Python讀取CSV文件UnicodeDecodeError終極解決方法
    在試圖使用已內置的CSV模塊讀取Python中時,CSV文件中的Unicode Decode Decode Decode Decode decode Error讀取,您可能會遇到錯誤的錯誤:無法解碼字節 in position 2-3: truncated \UXXXXXXXX escapeThi...
    程式設計 發佈於2025-05-22
  • 反射動態實現Go接口用於RPC方法探索
    反射動態實現Go接口用於RPC方法探索
    在GO 使用反射來實現定義RPC式方法的界面。例如,考慮一個接口,例如:鍵入myService接口{ 登錄(用戶名,密碼字符串)(sessionId int,錯誤錯誤) helloworld(sessionid int)(hi String,錯誤錯誤) } 替代方案而不是依靠反射...
    程式設計 發佈於2025-05-22
  • HTML格式標籤
    HTML格式標籤
    HTML 格式化元素 **HTML Formatting is a process of formatting text for better look and feel. HTML provides us ability to format text without us...
    程式設計 發佈於2025-05-22
  • Python中嵌套函數與閉包的區別是什麼
    Python中嵌套函數與閉包的區別是什麼
    嵌套函數與python 在python中的嵌套函數不被考慮閉合,因為它們不符合以下要求:不訪問局部範圍scliables to incling scliables在封裝範圍外執行範圍的局部範圍。 make_printer(msg): DEF打印機(): 打印(味精) ...
    程式設計 發佈於2025-05-22
  • Java中假喚醒真的會發生嗎?
    Java中假喚醒真的會發生嗎?
    在Java中的浪費喚醒:真實性或神話? 在Java同步中偽裝喚醒的概念已經是討論的主題。儘管存在這種行為的潛力,但問題仍然存在:它們實際上是在實踐中發生的嗎? Linux的喚醒機制根據Wikipedia關於偽造喚醒的文章,linux實現了pthread_cond_wait()功能的Linux實現,...
    程式設計 發佈於2025-05-22
  • 如何克服PHP的功能重新定義限制?
    如何克服PHP的功能重新定義限制?
    克服PHP的函數重新定義限制在PHP中,多次定義一個相同名稱的函數是一個no-no。嘗試這樣做,如提供的代碼段所示,將導致可怕的“不能重新列出”錯誤。 但是,PHP工具腰帶中有一個隱藏的寶石:runkit擴展。它使您能夠靈活地重新定義函數。 runkit_function_renction_...
    程式設計 發佈於2025-05-22
  • 如何使用Python理解有效地創建字典?
    如何使用Python理解有效地創建字典?
    在python中,詞典綜合提供了一種生成新詞典的簡潔方法。儘管它們與列表綜合相似,但存在一些顯著差異。 與問題所暗示的不同,您無法為鑰匙創建字典理解。您必須明確指定鍵和值。 For example:d = {n: n**2 for n in range(5)}This creates a dict...
    程式設計 發佈於2025-05-22

免責聲明: 提供的所有資源部分來自互聯網,如果有侵犯您的版權或其他權益,請說明詳細緣由並提供版權或權益證明然後發到郵箱:[email protected] 我們會在第一時間內為您處理。

Copyright© 2022 湘ICP备2022001581号-3