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

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

返回總目錄 本小節目錄 Pull Up Field(欄位上移) Pull Up Method(函數上移) Pull Up Constructor Body(構造函數本體上移) 1Pull Up Field(欄位上移) 概要 兩個子類擁有相同的欄位。將該欄位移至基類。 動機 如果各個子類是分別開發的, ...


返回總目錄

本小節目錄

1Pull Up Field(欄位上移)

概要

兩個子類擁有相同的欄位。將該欄位移至基類。

動機

如果各個子類是分別開發的,或者是在重構過程中組合起來的,你常常會發現它們擁有重覆性,特別是欄位容易重覆。判斷若幹欄位是否重覆,唯一的辦法就是觀察函數如何使用它們。如果它們被使用的方式很相似,就可以將它們歸納到基類中去。

範例

如下代碼所示,Employee的兩個子類Salesman和Enginner都有_name欄位,所以可以考慮把這個欄位提到基類中。

class Employee
{

}

class Salesman : Employee
{
    private string _name;
}

class Enginner : Employee
{
    private string _name;
}

重構後的代碼如下,這樣提的前提是這些子類有一個基類或者有很多相似的欄位和方法,不然為了一個欄位而單獨建立一個抽象類是不可取的,所以這個就需要具體權衡。

class Employee
{
    protected string _name;
}

class Salesman : Employee
{
 
}

class Enginner : Employee
{

}

小結

本項重構主要是減少重覆:首先它去除了重覆的數據聲明;其次它使你可以將該欄位的行為從子類移至基類,從而去除重覆的行為。

2Pull Up Method(函數上移)

概要

有些函數,在各個子類中產生完全相同的結果。將該函數移至基類。

動機

避免行為重覆是很重要的。儘管重覆的兩個函數也可以各自工作得很好,但重覆自身只會成為錯誤的滋生地,此外別無價值。無論何時,只要系統中出現重覆,你就面臨“修改其中一個卻未能修改另一個”的風險。

使用本項重構的場合:(1)如果某個函數在各個子類中的函數體都相同;(2)子類的函數覆寫了基類的函數,但卻仍然做相同的工作。

範例

以Customer表示“顧客”,它有兩個子類:表示“普通顧客”的RegularCustomer和表示“貴賓”的PreferredCustomer。

public abstract class Customer
{
    protected DateTime _lastBillDate;
    public void AddBill(DateTime date, double amount)
    {

    }
}

class RegularCustomer : Customer
{
    void CreateBill(DateTime date)
    {
        double chargeAmount = ChargeFor(_lastBillDate, date);
        AddBill(date, chargeAmount);
    }

    public double ChargeFor(DateTime start, DateTime end)
    {
        return 0;
    }
}

class PreferredCustomer : Customer
{
    void CreateBill(DateTime date)
    {
        double chargeAmount = ChargeFor(_lastBillDate, date);
        AddBill(date, chargeAmount);
    }

    public double ChargeFor(DateTime start, DateTime end)
    {
        return 100;
    }
}

兩個子類中都有一個CreateBill()函數,並且代碼完全一樣,但我不能直接把這個函數上移到基類中,因為各個子類的ChargeFor()函數並不相同。必須先在基類中聲明一個ChargeFor()抽象函數:

public abstract class Customer
{
    protected DateTime _lastBillDate;
    protected void AddBill(DateTime date, double amount)
    {

    }
    protected void CreateBill(DateTime date)
    {
        double chargeAmount = ChargeFor(_lastBillDate, date);
        AddBill(date, chargeAmount);
    }
    public abstract double ChargeFor(DateTime start, DateTime end);
}

class RegularCustomer : Customer
{
    public override double ChargeFor(DateTime start, DateTime end)
    {
        return 0;
    }
}

class PreferredCustomer : Customer
{
    public override double ChargeFor(DateTime start, DateTime end)
    {
        return 100;
    }
}

小結

這個重構要根據具體情況使用,如果不是每個子類都有這個方法的話,可以考慮使用介面或者其他方式。

3Pull Up Constructor Body(構造函數本體上移)

概要

你在各個子類中擁有一些構造函數,它們的本體幾乎完全一致。

在基類中新建一個構造函數,併在子類構造函數中調用它。

動機

如果你看見各個子類中的函數有共同行為,第一個念頭應該是將共同行為提煉到一個獨立函數中,然後將這個函數提升到基類。對於構造函數而言,它們彼此的共同行為往往就是“對象的建構”。這時候你需要在基類中提供一個構造函數,然後讓子類都來調用它。

範例

class Employee
{
    protected string _name;

    protected string _id;

}

class Manager:Employee
{
    private int _grade;
    public Manager(string name,string id,int grade)
    {
        _name = name;
        _id = id;
        _grade = grade;
    }
}

Employee的欄位應該在Employee構造函數中設值。因此定義了一個Employee構造函數,並將它聲明為Protected,表示子類應該調用它:

class Employee
{
    protected string _name;

    protected string _id;

    protected Employee(string name, string id)
    {
        _name = name;
        _id = id;
    }
}

class Manager : Employee
{
    private int _grade;
    public Manager(string name, string id, int grade) : base(name, id)
    {
        _grade = grade;
    }
}

小結

這個重構手法和提升欄位、提升方法很相似。只不過是將“對象的建構”提升到基類中。

 

To Be Continued……


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

-Advertisement-
Play Games
更多相關文章
  • qt學習教程1.qt開發環境搭建 首先下載qt 下載地址:http://download.qt.io/archive/qt/ 此教程使用的版本為5.1.1 下載好後,打開安裝包,然後點下一步 選擇一個位置來安裝qt,系統盤不夠用的就裝在其他盤 選擇同意 然後接下來一路確定下一步。就安裝好了。 第一個 ...
  • 凡是可以由自己命名的地方都稱為修飾符. 例: 項目名 ,包名 ,類名 .方法名 2. 命名規範. ① 不可使用java關鍵字和保留字,但是可以包含關鍵字和保留字. ② 可以使用26個字母大小寫,數字0-9,$和_. ③ 可以使用數字,但不可放在首位. ④ 長度理論上沒有限制,但命名最好能反映出其作用 ...
  • 1 s = input("輸入:") 2 result = '' 3 for i in range(len(s)): 4 result += chr(ord(s[i])^2000) 5 print(result) ord(char) #把字元轉換成unicode編碼(整型) chr(unicode) ...
  • inspect模塊用於收集python對象的信息,可以獲取類或函數的參數的信息,源碼,解析堆棧,對對象進行類型檢查等等,有幾個好用的方法: getargspec(func) 返回一個命名元組ArgSpect(args, varargs, keywords, defaults),args是函數位置參數 ...
  • python數據轉換json 將json轉換為pathon數據 repr 和 eval用法 json讀取和寫入 總結: 數據轉換 第一步: 引入json包: import json 第二步: 使用 json.dumps(pythonObj) 把python數據轉換json數據 第三步: 使用json ...
  • 許久沒更新博客了! spring還有一章aop(面向切麵),我就沒講述了,你們可以去看下代理模式。 那麼我們開始整合:struts2 2.3.4 ,hibernate 5.2.10 ,spring 4.3.10 ,一直以來用到的xml式,那麼整合也不例外,就是有些麻煩。另外註解式想瞭解請留言(雖然s ...
  • 模擬用戶登錄 ...
  • 整合SSH時,遇到了org.springframework.beans.factory.BeanCreationException錯誤 ...
一周排行
    -Advertisement-
    Play Games
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...