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

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

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

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

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刪除
最新教學 更多>
  • 如何在 PHP 中組合兩個關聯數組,同時保留唯一 ID 並處理重複名稱?
    如何在 PHP 中組合兩個關聯數組,同時保留唯一 ID 並處理重複名稱?
    在 PHP 中組合關聯數組在 PHP 中,將兩個關聯數組組合成一個數組是常見任務。考慮以下請求:問題描述:提供的代碼定義了兩個關聯數組,$array1 和 $array2。目標是建立一個新陣列 $array3,它合併兩個陣列中的所有鍵值對。 此外,提供的陣列具有唯一的 ID,而名稱可能重疊。要求是建...
    程式設計 發佈於2024-12-29
  • 大批
    大批
    方法是可以在物件上呼叫的 fns 數組是對象,因此它們在 JS 中也有方法。 slice(begin):將陣列的一部分提取到新數組中,而不改變原始數組。 let arr = ['a','b','c','d','e']; // Usecase: Extract till index ...
    程式設計 發佈於2024-12-29
  • 在 Go 中使用 WebSocket 進行即時通信
    在 Go 中使用 WebSocket 進行即時通信
    构建需要实时更新的应用程序(例如聊天应用程序、实时通知或协作工具)需要一种比传统 HTTP 更快、更具交互性的通信方法。这就是 WebSockets 发挥作用的地方!今天,我们将探讨如何在 Go 中使用 WebSocket,以便您可以向应用程序添加实时功能。 在这篇文章中,我们将介绍: WebSoc...
    程式設計 發佈於2024-12-29
  • 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...
    程式設計 發佈於2024-12-29
  • 插入資料時如何修復「常規錯誤:2006 MySQL 伺服器已消失」?
    插入資料時如何修復「常規錯誤:2006 MySQL 伺服器已消失」?
    插入記錄時如何解決「一般錯誤:2006 MySQL 伺服器已消失」介紹:將資料插入MySQL 資料庫有時會導致錯誤「一般錯誤:2006 MySQL 伺服器已消失」。當與伺服器的連線遺失時會出現此錯誤,通常是由於 MySQL 配置中的兩個變數之一所致。 解決方案:解決此錯誤的關鍵是調整wait_tim...
    程式設計 發佈於2024-12-29
  • Bootstrap 4 Beta 中的列偏移發生了什麼事?
    Bootstrap 4 Beta 中的列偏移發生了什麼事?
    Bootstrap 4 Beta:列偏移的刪除和恢復Bootstrap 4 在其Beta 1 版本中引入了重大更改柱子偏移了。然而,隨著 Beta 2 的後續發布,這些變化已經逆轉。 從 offset-md-* 到 ml-auto在 Bootstrap 4 Beta 1 中, offset-md-*...
    程式設計 發佈於2024-12-29
  • 儘管程式碼有效,為什麼 POST 請求無法擷取 PHP 中的輸入?
    儘管程式碼有效,為什麼 POST 請求無法擷取 PHP 中的輸入?
    解決PHP 中的POST 請求故障在提供的程式碼片段中:action=''而非:action="<?php echo $_SERVER['PHP_SELF'];?>";?>"檢查$_POST陣列:表單提交後使用 var_dump 檢查 $_POST 陣列的內...
    程式設計 發佈於2024-12-29
  • 如何在 Python 中檢查 NaN(不是數字)?
    如何在 Python 中檢查 NaN(不是數字)?
    檢查 NaN(不是數字)在 Python 中,NaN(不是數字)由 float('nan') 表示。它用於表示無法表示為實數的值。若要檢查值是否為 NaN,請使用 math.isnan 函數。 範例:import math x = float('nan') if math.isnan...
    程式設計 發佈於2024-12-29
  • 如何在 React 中有條件地應用類別屬性?
    如何在 React 中有條件地應用類別屬性?
    在React 中有條件地應用類別屬性在React 中,根據從父組件傳遞的props 來顯示或隱藏元素是很常見的。為此,您可以有條件地應用 CSS 類別。然而,當使用語法 {this.props.condition ? 'show' : 'hidden'} 直接在字串中...
    程式設計 發佈於2024-12-28
  • 如何在Java中執行系統命令並與其他應用程式互動?
    如何在Java中執行系統命令並與其他應用程式互動?
    Java 中運行進程在 Java 中,啟動進程的能力是執行系統命令和與其他應用程式互動的關鍵功能。為了啟動一個流程,Java提供了一個相當於.Net System.Diagnostics.Process.Start方法。 解決方案:取得本地路徑對於執行至關重要Java 中的程序。幸運的是,Java ...
    程式設計 發佈於2024-12-28
  • 如何在 C++ 中建立多行字串文字?
    如何在 C++ 中建立多行字串文字?
    C 中的多行字串文字 在 C 中,定義多行字串文字並不像 Perl 等其他語言那麼簡單。但是,您可以使用一些技術來實現此目的:連接字串文字一種方法是利用 C 中相鄰字串文字由編譯器連接的事實。將字串分成多行,您可以建立單一多行字串:const char *text = "This te...
    程式設計 發佈於2024-12-28
  • 如何準確地透視具有不同記錄的資料以避免遺失資訊?
    如何準確地透視具有不同記錄的資料以避免遺失資訊?
    有效地透視不同記錄透視查詢在將資料轉換為表格格式、實現輕鬆資料分析方面發揮著至關重要的作用。但是,在處理不同記錄時,資料透視查詢的預設行為可能會出現問題。 問題:忽略不同值考慮下表:------------------------------------------------------ | Id...
    程式設計 發佈於2024-12-27
  • 為什麼 C 和 C++ 忽略函式簽章中的陣列長度?
    為什麼 C 和 C++ 忽略函式簽章中的陣列長度?
    將陣列傳遞給C 和C 中的函數問題:為什麼C和C 編譯器允許在函數簽章中宣告數組長度,例如int dis(char a[1])(當它們不允許時)強制執行? 答案:C 和C 中用於將數組傳遞給函數的語法是歷史上的奇怪現象,它允許將指針傳遞給第一個元素詳細說明:在C 和C 中,數組不是透過函數的引用傳遞...
    程式設計 發佈於2024-12-26
  • 如何刪除 MySQL 中的重音符號以改進自動完成搜尋?
    如何刪除 MySQL 中的重音符號以改進自動完成搜尋?
    在MySQL 中刪除重音符號以實現高效的自動完成搜尋管理大型地名資料庫時,確保準確和高效至關重要資料檢索。使用自動完成功能時,地名中的重音可能會帶來挑戰。為了解決這個問題,一個自然的問題出現了:如何在 MySQL 中刪除重音符號以改善自動完成功能? 解決方案在於為資料庫列使用適當的排序規則設定。透過...
    程式設計 發佈於2024-12-26
  • 如何在MySQL中實作複合外鍵?
    如何在MySQL中實作複合外鍵?
    在 SQL 中實作複合外鍵一個常見的資料庫設計涉及使用複合鍵在表之間建立關係。複合鍵是多個列的組合,唯一標識表中的記錄。在這個場景中,你有兩個表,tutorial和group,你需要將tutorial中的複合唯一鍵連結到group中的欄位。 根據MySQL文檔,MySQL支援外鍵對應到複合鍵。但是,...
    程式設計 發佈於2024-12-26

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

Copyright© 2022 湘ICP备2022001581号-3