重構手法之處理概括關係【3】

来源:http://www.cnblogs.com/liuyoung/archive/2017/12/08/7967041.html
-Advertisement-
Play Games

返回總目錄 本小節目錄 Extract BaseClass(提煉基類) Extract Interface(提煉介面) Collapse Hierarchy(摺疊繼承體系) 7Extract BaseClass(提煉基類) 概要 兩個類有相似特性。為這兩個類建立一個基類,將相同特性移至基類。 動機 ...


返回總目錄

本小節目錄

7Extract BaseClass(提煉基類)

概要

兩個類有相似特性。為這兩個類建立一個基類,將相同特性移至基類

動機

重覆代碼是系統中最糟糕的東西之一。如果你在不同地方做同一件事情,一旦需要修改那些動作,你就得平白做更多的修改。

重覆代碼的某種形式就是:兩個類以相同的方式做類似的事情,或者以不同的方式做類似的事情。對象提供了一種簡化這種情況的機制,那就是繼承。但是,在建立這些具有共通性的類之前,你往往無法發現這樣的共通性,因此常常會在具有共通性的類出現之後,再開始建立其間的繼承結構。

範例

Dog類中的EatFood和Groom有可能被其他類用到,因為他們都是動物的一些公有性質,所以這個時候我們就會考慮對它進行提煉。

public class Dog
{
    public void EatFood()
    {
        // eat some food
    }


    public void Groom()
    {
        // perform grooming
    }
}

代碼如下所示,提取了Animal方法來封裝公用的EatFood和Groom類,從而使其他繼承了Animal類的子類都可以使用這兩個方法了。

public class Animal
{
    public void EatFood()
    {
        // eat some food
    }

    public void Groom()
    {
        // perform grooming
    }
}

public class Dog : Animal
{
}

小結

這個重構是典型的繼承用法,很多程式員都會選擇這樣做,但是要註意正確的使用,不要造成過度使用了繼承,如果過度使用了,請考慮用介面、組合和聚合來實現。

這個手法也經常用到,比如做的MVC項目中的BaseController。

8Extract Interface(提煉介面)

概要

若幹客戶使用類介面中的同一子集,或者兩個類的介面有部分相同。將相同的子集提煉到一個獨立介面中。

動機

如果某個類在不同環境下扮演截然不同的角色,使用介面就是個好主意。你可以針對每個角色以Extract Interface提煉出相應介面。另一種可以用Extract Interface的情況是:你想要描述一個類的外部依賴介面(outbound interface,即這個類要求服務提供方提供的操作)。如果你打算將來加入其它種類的服務對象。只需要求它們實現這個介面即可。

範例

TimeSheet類表示員工為客戶工作的時間表,從中可以計算客戶應該支付的費用。為了計算這些費用,TimeSheet需要知道員工級別,以及該員工是否具有特殊技能:

class TimeSheet
{
    public double Charge(Employee emp, int days)
    {
        int baseCharge = emp.GetRate() * days;
        if (emp.HasSpecialSkill())
        {
            return baseCharge * 1.05;
        }
        return baseCharge;
    }
}

class Employee
{
    public int GetRate()
    {
        return 2;
    }

    public bool HasSpecialSkill()
    {
        return true;
    }
}

除了提供員工的級別和特殊技能信息之外,Employee還有很多其他方面的功能,但本應用程式只需這兩項功能。可以針對這兩項功能定義一個介面,從而強調“我只需要這部分功能”的事實:

public interface IBillable
{
    int GetRate();
    bool HasSpecialSkill();
}

然後,聲明讓Employee實現這個介面:

class Employee : IBillable
{
    public int GetRate()
    {
        return 2;
    }

    public bool HasSpecialSkill()
    {
        return true;
    }
}

完成以後,修改TimeSheet類中Charge()函數聲明,強調該函數只使用Employee的這一部分行為:

class TimeSheet
{
    public double Charge(IBillable emp, int days)
    {
        int baseCharge = emp.GetRate() * days;
        if (emp.HasSpecialSkill())
        {
            return baseCharge * 1.05;
        }
        return baseCharge;
    }
}

到目前為止,我們只不過是在文檔化方面有一點收穫。但就這一個函數而言,這樣的收穫並沒有太大的價值;但如有若幹個類都使用IBillable介面,它就會很有用。如果我還想計算電腦租金,巨大的收穫就顯露出來了;要想計算客戶租用電腦的費用,只需讓Computer類實現IBillable介面,然後就可以把租用電腦的時間也填到時間表上了。

小結

這個重構策略也是一個常見的運用,很多設計模式也會在其中運用此思想。

9Collapse Hierarchy(摺疊繼承體系)

概要

基類和子類之間無太大區別。將它們合為一體。

動機

如果你曾經編寫過繼承體系,就會知道,繼承體系很容易變得過分複雜。所謂重構繼承體系,往往是將函數和欄位在體系中上下移到。完成這些動作後,你很可能發現某個子類並未帶來該有的價值,因此需要把基類和子類合併起來。

範例

如下代碼所示,StudentWebSite子類除了有一個屬性用來說明網站是否是活動的外沒有別的責任,在這種情形下我們意識到IsActive屬性可以應用到所有的網站,所以我們可以將IsActive屬性上移到基類中,並去掉StudentWebSite類。

public class Website
{
    public string Title { get; set; }
    public string Description { get; set; }
    public IEnumerable<Webpage> Pages { get; set; }
}

public class StudentWebsite : Website
{
    public bool IsActive { get; set; }
}

public class Webpage
{
}

重構後的代碼如下:

public class Website
{
    public string Title { get; set; }
    public string Description { get; set; }
    public IEnumerable<Webpage> Pages { get; set; }
    public bool IsActive { get; set; }
}
public class Webpage
{
}

小結

這項重構和前幾篇最主要論述了子類和父類的繼承關係以及如何判斷什麼時候需要使用繼承,一般我們都能處理好這些關係,所以相對比較簡單。

 

To Be Continued……


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

-Advertisement-
Play Games
更多相關文章
  • 1 # coding:utf-8 2 import os 3 # 聲明一個空列表存儲學員的姓名 4 member_list = [] 5 # 讀取本地文件生數據,將本地數據添加至member_list 中 即可沒保證第二次運行member_list 6 if os.path.exists('stud... ...
  • 編寫Python網路爬蟲,爬取新浪新聞首頁新聞列表。有代碼,有詳細步驟,有截圖,歡迎嘗試! ...
  • 許久不見的朋友請我吃飯,期間給我介紹他一個弟弟,說明年要畢業了,還不知道找啥工作,說有培訓機構讓他學VR、大數據什麼的,不知道前景咋樣,想咨詢一下我。相信很多朋友面臨畢業,都不知道該從事哪個行業,自己又有能力做哪些工作,很迷茫!他說你早九晚六,周末雙休,工作輕鬆工資又不低,多好啊,能不能讓弟弟也乾你 ...
  • 一、前言 隨著每天業務的增長,Tomcat 的catalina.out日誌 變得越來越大,占用磁碟空間不說。要查看某個時候的日誌的時候,龐大的日誌讓你頓時無從下手,所以日誌的切割的變得刻不容緩。而且,切割後的日誌,還可以定期清理掉久遠的日誌...... 二、Tomcat 日誌分割 我們採用日期形式切 ...
  • 1、官網:http://www.python.org/download/下載安裝包,目前最新版本為3.6,安裝包很多地方可以下,也可以在360軟體管家上下載安裝 特別要註意勾選:Add Python 3.5 to PATH,這樣就可以不用配置環境變數了,然後選擇立即安裝 2、也可手動進行配置環境變數 ...
  • 學習java也一個月多了,我這個人吧總感覺格局有點小,太小家子氣,在學習新東西的過程中我開始逐漸對自己有一些認識吧。首先就是學習的進度不夠快,總是執著與基礎,就像高中的時候吧,雖然當時在我們班我的成績足夠好了,但自己學習複習的過程中仍然執著與基礎。導致自己的學習進度不快,對於自己來說也限制了自己更多 ...
  • 基本概念 java學習群669823128 操作系統中 heap 和 stack 的區別 什麼是基於註解的切麵實現 什麼是 對象/關係 映射集成模塊 什麼是 Java 的反射機制 什麼是 ACID BS與CS的聯繫與區別 Cookie 和 Session的區別 fail-fast 與 fail-sa ...
  • C#有一個關鍵詞yield,簡化遍歷操作實現的語法糖。 下麵Insus.NET使用例子來說明,就拿昨晚的一篇《從字元串數組中把數字的元素找出來》 http://www.cnblogs.com/insus/p/8001026.html 其中有一段控制台程式: 上圖中的#17行代碼至#29行代碼。按照面 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...