3種方式幫助我們修改代碼

来源:http://www.cnblogs.com/LipeiNet/archive/2016/01/06/5103158.html
-Advertisement-
Play Games

在實際的項目開發中我們可能經常要修改已有的代碼,可能我們經常說開閉原則對已有的代碼不准修改,但是實際上很難,那麼下麵的3種方法也許能幫助我們改善對代碼的修改。1:新生方法有時候在我們開發的時候需要像系統加入新的功能時候這個時候我們就可能改變我們原有方法的結構。那麼下麵有一個簡單例子以前添加用戶的.....


    在實際的項目開發中我們可能經常要修改已有的代碼,可能我們經常說開閉原則對已有的代碼不准修改,但是實際上很難,那麼下麵的3種方法也許能幫助我們改善對代碼的修改。

1:新生方法

有時候在我們開發的時候需要像系統加入新的功能時候這個時候我們就可能改變我們原有方法的結構。那麼下麵有一個簡單例子

以前添加用戶的業務

public void Add(IList<UserInfo> userInfos)
        {        
            if (userInfos!=null)
            {
                Controller.Add(userInfos);
            }
        }

現在用戶分級別,不同級別的存入不同的表,下麵是白金用戶的添加

public void Add(IList<UserInfo> userInfos)
        {        
            if (userInfos!=null)
            {
                IList<UserInfo> mUserInfos=userInfos.ToList().FindAll(p=>p.Type=="白金").ToList();
                Controller.Add(userInfos);
            }
        }

很明顯這種寫法導致新老代碼混合在一起,我們沒法對新的代碼進行測試操作。那麼我們就可以新生一個方法

public void Add(IList<UserInfo> userInfos)
        {
            if (userInfos != null)
            {
                userInfos = GetUserInfosByType(userInfos, "白金");//此處引用
                Controller.Add(userInfos);
            }
        }
        /// <summary>
        /// 新生的方法
        /// </summary>
        protected virtual IList<UserInfo> GetUserInfosByType(IList<UserInfo> userInfos, string type)
        {
            return userInfos.ToList().FindAll(p => p.Type == type).ToList();
        }

這就是一個新生方法的例子。以下使我們採取的步驟:

1:確定需要修改的地方

2:確定新生方法需要向源方法返回的值(如果需要的話)

3:把新生的方法採用受保護的虛方法,因為這樣可以加入測試(如果條件不允許的話就可以放棄對老代碼的測試)

優點:新老代碼隔離,即使是無法對老代碼進行測試,那麼新加上的代碼可以進行測試。

缺點:暫時放棄了老的代碼,你基本不在為他編寫測試也不會對他進行改善。

2:包裝方法

這個和上面一個很相似。同樣我們需要向一個現存的方法添加行為,這個很容易實現,但是如果我們的項目本身沒有單元測試這個就變得很危險。在早期的設計我們的目的肯定是做一個單一的功能,但是由於業務的需要導致這個現存的方法變得越來越膨脹,而且耦合度極高,沒有接縫,那麼我們就無法進行分離。所以我們可以採取包裝方法的模式來應對業務的改變

比喻我們原來的代碼只是添加員工(下麵只是簡單例子,實際肯定比這複雜)

public void AddEmployee(UserInfo userInfo)
        {
....... Controller.Add(userInfo);
....... }

但是現在對於重要的信息加入操作日誌所以我採取把原來的老方法包裝成Add如下

public void AddEmployee(UserInfo userInfo)
        {
            Add(userInfo);
            AddLog(userInfo);//日誌操作
        }

        public void Add(UserInfo userInfo)
        {
...... Controller.Add(userInfo);
...... }

我們使用包裝方式的步驟:

1:確實要修改的地方

2:修改舊方法的名字

3:在新方法調用舊方法和其他需要調用的方法

缺點:

1:新添加的邏輯無法與以前舊的方法緊密聯合在一起,在他們之前或者之

2:需要為舊方法重新命名(這個很難)可能導致代碼的可讀性變得很差

優點:

1:新的代碼變得可測

2:在引入接縫的同時加入了添的邏輯

 3:包裝類

其實包裝類採用的是組合模式,把原來的舊的類重新進行包裝,然後導入新添加行為的類中,這樣就可以添加新的行為但是卻沒有破壞舊類的行為。舉一個簡單例子

public class LogEmplyee
    {
        private Employee e;

        public LogEmplyee(Employee employee)
        {
            e = employee;
        }

        public void AddEmployee(UserInfo userInfo)
        {
            e.Add(userInfo);
            AddLog(userInfo);
        }
    }

其實這採用也是我們經常說的裝飾模式。

什麼情況下我們使用包裝類

1:我們舊的類已經很大了,實在不想讓他繼續膨脹

2:我們添加新的行為很多,並且大多使用了舊類中的方法,並且有先後順序,我們就可以考慮

小結:通過上面三種方式,我們可以使他們來修改代碼,而不需要為現存的類編寫測試,當時從設計的角度來考慮確實不是很好理解,但是這麼做起碼讓我們來改善我們的設計了,我們把新方法和舊方法進行了隔離,我們留下了接縫以後後期需要測試我們可以加入測試,來為我們的代碼提供保障。慢慢一點點的改變相信會使代碼變得更好。

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • C++中的const可用於修飾變數、函數,且在不同的地方有著不同的含義,現總結如下。const的語義C++中的const的目的是通過編譯器來保證對象的常量性,強制編譯器將所有可能違背const對象的常量性的操作都視為error。對象的常量性可以分為兩種:物理常量性(即每個bit都不可改變)和邏輯常量...
  • 一,java.util.regex包中提供了兩個類來表示對正則表達式的支持1.Matcher,通過解釋Pattern對character sequence 執行匹配操作的引擎public final class Matcher implements MatchResult2.Pattern,正則表達...
  • 1.1 框架的概念框架其實就是可重用代碼的集合,框架的代碼是框架架構的代碼,不是業務邏輯代碼,框架代碼保護類.方法.函數等等,框架代碼按照一定的規則組合起來就形成了框架。1.2 不使用框架開發的時候遇到的問題 1.代碼編寫沒有統一的規範 2.項目功能不能很好的拆分 3.一個局部的微小改動可能會...
  • 五種I/O: 1)阻塞I/0 2)非阻塞I/O 3)I/O復用 4)事件(信號)驅動I/O 5)非同步I/O
  • 接著上一篇博客繼續Tomcat配置。3. 虛擬目錄映射虛擬目錄是與實際目錄相對應的,不是一個實際存在的目錄。配置虛擬目錄有兩點好處:1、 便於理解;2、如果web應用所在目錄更改,只需要更改虛擬目錄對應的實際目錄,而外界仍可以通過原方式訪問新的web應用。Tomcat 中配置虛擬目錄有以下三中方式:...
  • 前面我們已經接觸過幾種數據結構了,有數組、鏈表、Hash表、紅黑樹(二叉查詢樹),今天再來看另外一種數據結構:棧。 什麼是棧呢,我就不找它具體的定義了,直接舉個例子,棧就相當於一個很窄的木桶,我們往木桶里放東西,往外拿東西時會發現,我們最開始放的東西在最底部,最先拿出來的是剛剛放進去的。所以,...
  • Sample.ashx(handler): using System;using System.Collections.Generic;using System.Web;using System.IO;public ...
  • 為什麼要對資料庫進行設計? 當我們要存儲的數據比較少的是後當然不需要對資料庫進行設計,但是當我們對數據的需求量越來越大時對資料庫的設計就很有必要性了!如果資料庫的設計不當會造成數據冗餘、修改複雜、操作數據異常等問題而好的資料庫設計則可以減少不必要的數據冗餘,通過合理的數據規劃提高系統的性能! 什麼是...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...