重構手法之重新組織函數【4】

来源:http://www.cnblogs.com/liuyoung/archive/2017/11/20/7846156.html
-Advertisement-
Play Games

返回總目錄 本小節目錄 Split Temporary Variable(分解臨時變數) Remove Assignments to Parameters(移除對參數的賦值) Remove Assignments to Parameters(移除對參數的賦值) 6 Split Temporary V ...


返回總目錄

本小節目錄

6 Split Temporary Variable(分解臨時變數)

概要

你的程式有某個臨時變數被賦值超過一次,它既不是迴圈變數,也不被用於收集計算結果。

針對每次賦值,創造一個獨立、對應的臨時變數。

動機

臨時變數有各種不同的用途。

1、迴圈變數;

2、結果收集變數;

3、保存一段冗長代碼的運算結果,便於稍後使用。

其中第三種情況的臨時變數應該只被賦值一次。如果它們被賦值超過一次,就意味著它們在函數中承擔了一個以上的責任。如果臨時變數承擔了多個責任,它就應該被分解為多個臨時變數,每個變數只承擔一個責任。同一個臨時變數承擔兩件不同的事情,會令代碼閱讀者糊塗。

範例

double GetTotalCost()
{
    double result = 0;

    double money = _chickMoney + _chipMoney;

    result += money;

    money = _cocoaMoney + _coffeeMoney;

    result += money;

    return result;
}

可以看到在這個範例中,臨時變數money被賦值兩次。並且它沒有做到累積結果的作用,累積結果給了result。所以,我們需要做重構,讓這個變數的意圖變的更加明確。

所以我們第一步,尋找這個變數第一次聲明的地方,並且將他改名,然後修改在第二次賦值之前的所有引用點,並且在第二次賦值處進行重新聲明:

double GetTotalCost()
{
    double result = 0;

    double mealMoney = _chickMoney + _chipMoney;

    result += mealMoney ;

    double money = _cocoaMoney + _coffeeMoney;

    result += money;

    return result;
}

現在,新的臨時變數只承擔原先money的第一個責任。而且我們在原先money變數第二次被賦值處重新聲明瞭money。然後,繼續處理money臨時變數的第二次賦值。

double GetTotalCost()
{
    double result = 0;

    double mealMoney = _chickMoney + _chipMoney;

    result += mealMoney ;

    double drinkMoney = _cocoaMoney + _coffeeMoney;

    result += drinkMoney ;

    return result;
}

可以看到,我們完成了變數的重構之後,函數對於臨時變數之前的money的困惑已經沒有了,整體因為變數名字的本身使得邏輯更加清晰。

如果在這裡你的代碼還是比較複雜的話,可以盡情使用其他的重構手法。

 小結

這個重構手法的重點在於:臨時變數不是用於迴圈變數和結果收集,但卻被賦值超過兩次,那就對它進行分解,使其每次只承擔一個責任。

7 Remove Assignments to Parameters(移除對參數的賦值)

概要

代碼對一個參數進行賦值。

以一個臨時變數取代該參數的位置。

動機

首先要明確這裡“對參數賦值”的意思。如果你把一個名為foo的對象作為參數傳給某個函數,那麼“對參數賦值”就意味著改變foo,使它引用另一個對象。如果在“被傳入對象”身上進行什麼操作,那是沒問題的。Java只採用按值傳遞的方式,而C#分為值傳遞和引用傳遞,關於C#的值傳遞和引用傳遞,請看我的另一篇文章,或者自行百度。

int test(int a)
{
    if (a > 50)
    {
        a = 1;
    }
    return a;
}

這個就違反了這個原則,因為你對傳入參數進行重新賦值會讓代碼閱讀者產生歧義,降低了代碼的清晰度。他們搞不清甚至看不懂你參數到底代表什麼含義,甚至會對你這個參數的穩定性表示擔憂。

在值類型按值傳遞的情況下,對參數的任何修改,都不會對調用端造成任何影響。這個重構手法也是針對值類型按值傳遞的。

範例

int GetDiscount(int inputVal, int quantity, int yearToDate)
{
     if (inputVal > 50)
     {
         inputVal -= 2;
     }
     if (quantity > 100)
     {
         inputVal -= 1;
     }
     if (yearToDate > 1000)
     {
         inputVal -= 4;
     }
     return inputVal;
}

以臨時變數取代對參數的賦值動作,得到以下代碼:

int GetDiscount(int inputVal, int quantity, int yearToDate)
{
     int result=inputVal;
     if (inputVal > 50)
     {
         result-= 2;
     }
     if (quantity > 100)
     {
         result-= 1;
     }
     if (yearToDate > 1000)
     {
         result-= 4;
     }
     return result;
}

 小結

如果參數只表示“被傳遞進來的東西”,那麼代碼會很清晰。

 

To Be Continued...


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

-Advertisement-
Play Games
更多相關文章
  • Nginx是個輕量級的反向代理,當然,也有相應的SSL身份認證。本文主要採用一種自我寄宿的方式,使用Nginx集群,通過windows證書(X.509證書),講述客戶端如何訪問伺服器的方法。客戶端以BasicHttpBinding進行訪問Nginx,然後Nginx進行負載均衡,將消息分發到後端任意一... ...
  • 1,菜單。在最新的abp3.1.2中 菜單欄是在左側的如下圖(上中下的結構),中間部分就是我們要添加菜單的地方。 ABP集成了通用的創建和顯示菜單的方式,在展示層web下的appstart文件夾下找到AbpNavigationProvider 添加如下代碼即可 關於菜單部分的具體實現原理,可以在ht ...
  • 1 軟體度量值指標 1.1 可維護性指數 表示源代碼的可維護性,數值越高可維護性越好。該值介於0到100之間。綠色評級在20到100之間,表明該代碼具有高度的可維護性;黃色評級在10到19之間,表示該代碼適度可維護;紅色評級在0至9之間,表示低可維護性。 1.2 圈複雜度 它是通過計算程式流中不同代 ...
  • 當我們使用QQ的時候就會發現,他可以啟動多個QQ,但是有時候,我們不想這樣做,這時候我們就需要使用到單例模式. 1.將Form2的構造函數轉為私有 using System.Windows.Forms; namespace 單例模式 { public partial class Form2 : Fo ...
  • 演示產品下載地址:http://www.jinhusns.com ...
  • 下載地址:http://ilspy.net/ 中文版下載地址:http://www.fishlee.net/soft/ilspy_chs 對dll和exe文件反編譯: ...
  • 在Winform開發中,我們往往除了常規的單表信息錄入外,有時候設計到多個主從表的數據顯示、編輯等界面,單表的信息一般就是控制項和對象實體一一對應,然後調用API保存即可,主從表就需要另外特殊處理,本隨筆介紹如何快速實現主從表編輯界面的處理,結合GridControl控制項的GridView控制項對象,實... ...
  • 為了防止不提供原網址的轉載,特在這裡加上原文鏈接: "http://www.cnblogs.com/skabyy/p/7695258.html" 本篇將實現登錄、許可權控制、日誌配置與審計日誌的功能。首先我們先實現登錄功能,在登錄的基礎上,通過控權使得只有ID為1988的用戶才能創建tweet。最後配 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...