成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Java中的代碼重構:技巧、優秀實踐與方法

譯文 精選
開發 后端
Java 代碼重構是一種在不影響代碼外部行為的前提下進行的代碼優化,它通過漸進和小規模的優化來改善現有代碼的結構和質量。重構的目標是提高代碼的可讀性、性能、可維護性和效率等。

譯者 | 劉汪洋

審校 | 重樓

什么是 Java 代碼重構?

Java 代碼重構是一種在不影響代碼外部行為的前提下進行的代碼優化,它通過漸進和小規模的優化來改善現有代碼的結構和質量。重構的目標是提高代碼的可讀性、性能、可維護性和效率等。

Martin Fowler 是這個領域的權威的大牛和非常高產的作家,他在多篇文章和書籍中探討了代碼設計和重構的主題。在他的作品《重構:改善既有代碼的設計》中,他精辟地解釋了重構的本質:

_“重構是在不改變代碼外在行為的前提下,對代碼做出修改,以改進程序內在結構的過程。重構是一種經過千錘百煉形成的有條不紊的程序整理方法,可以最大限度地減少整理過程中引入的錯誤幾率。其核心是不斷進行一些小的優化,每個優化看似不起眼,但積少成多,效果顯著。”

——Martin Fowler

在進行 Java 代碼重構時,可以考慮以下常見的優化措施:

  • 為變量、類、函數和其他元素重命名,使其具有更好的可讀性和自描述性。
  • 通過內聯減少方法或函數調用,使用更簡潔的內容。
  • 抽取函數中的代碼塊,并將它們移到新的獨立函數中,以增強模塊性和可讀性。
  • 消除冗余,通過刪除相同功能的多個代碼片段,并將它們合并為一個。
  • 拆分處理過多職責的類或模塊,將其分解為更小、更有內聚的組件。
  • 合并具有相似功能的類或模塊,簡化結構。
  • 進行代碼性能方面的優化

Java 代碼重構技巧

變量和方法的重新命名

為變量和方法選擇具有代表性的名稱是增強代碼可讀性的重要方法。

代碼的可讀性是構建高質量代碼庫的關鍵要素之一。易讀的代碼能夠清晰地表達其目的,而難以理解的代碼則會增加重構過程中出現錯誤的風險。采用有意義的變量和方法名稱可以減少注釋的需求,并降低溝通成本。

// 重構前
int d = 30; // 天數
int h = 24; // 一天的小時數

// 重構后
int daysInMonth = 30;
int hoursInDay = 24;

方法的提取

在 Java 代碼重構技術中,方法的提取是一種常見而實用的策略。當一個方法變得過長和過于復雜時,通過提取部分功能到一個新的方法中能夠使原方法更簡潔和易讀。這不僅使代碼更具可維護性,還提高了其可重用性。

假設你有一個簡單的類,用于處理訂單并計算小計、稅費和總費用。

public class OrderProcessor {
    private List<Item> items;
    private double taxRate;

    public double processOrder() {
        double subtotal = 0;
        for (Item item : items) {
            subtotal += item.getPrice();
        }
        
        double totalTax = subtotal * taxRate;
        double totalCost = subtotal + totalTax;
        
        return totalCost;
    }
}

你可以將該代碼重構,把計算小計、稅費和總費用的代碼分別提取到 calculateSubtotal、calculateTax 和 calculateTotalCost 三個獨立的方法中,從而使類更加易讀、模塊化和可重用。

public class OrderProcessor {
    private List<Item> items;
    private double taxRate;

    public double processOrder() {
        double subtotal = calculateSubtotal();
        double totalTax = calculateTax(subtotal);
        return calculateTotalCost(subtotal, totalTax);
    }
    
    private double calculateSubtotal() {
        double subtotal = 0;
        for (Item item : items) {
            subtotal += item.getPrice();
        }
        return subtotal;
    }
    
    private double calculateTax(double subtotal) {
        return subtotal * taxRate;
    }
    
    private double calculateTotalCost(double subtotal, double totalTax) {
        return subtotal + totalTax;
    }
}

消除“魔法”數字和字符串

“魔法”數字和字符串是指直接硬編碼在代碼中的值。這種做法不僅會降低代碼的可維護性,還可能因輸入錯誤而導致結果不一致和錯誤的增多。為了避免這樣的問題,你應當避免使用硬編碼的值,而是通過使用具有清晰描述性的常量來重構你的代碼。

// 重構前
if (status == 1) {
    // ... 活躍狀態的代碼 ...
}

// 重構后
public static final int ACTIVE_STATUS = 1;

if (status == ACTIVE_STATUS) {
    // ... 活躍狀態的代碼 ...
}

代碼復用

代碼復用是指刪除代碼庫中多處出現的重復或相似的代碼段。這樣的代碼不僅降低了代碼質量和效率,還可能導致 bug 更加頻繁的出現和代碼庫變得更加復雜。因此,開發人員通常會對這類代碼感到反感。為了優化代碼,我們可以考慮提取重復部分來創建可復用的方法或函數,同時確保重構過程中保持原有代碼的功能和邏輯。

重構前

public class NumberProcessor {

    // 計算總和
    public int calculateTotal(int[] numbers) {
        int total = 0;
        for (int i = 0; i < numbers.length; i++) {
            total += numbers[i];
        }
        return total;
    }

    // 計算平均值
    public double calculateAverage(int[] numbers) {
        int total = 0;
        for (int i = 0; i < numbers.length; i++) {
            total += numbers[i];
        }
        double average = (double) total / numbers.length;
        return average;
    }
}

重構后

public class NumberProcessor {

     // 計算總和
    public int calculateSum(int[] numbers) {
        int total = 0;
        for (int i = 0; i < numbers.length; i++) {
            total += numbers[i];
        }
        return total;
    }

    // 計算總和
    public int calculateTotal(int[] numbers) {
        return calculateSum(numbers);
    }

    // 計算平均值
    public double calculateAverage(int[] numbers) {
        int total = calculateSum(numbers);
        double average = (double) total / numbers.length;
        return average;
    }
}

在優化后的代碼中,我們將用于計算數組總和的邏輯提取到了一個名為 calculateSum 的獨立方法中。現在,calculateTotal 和 calculateAverage 方法可以直接調用 calculateSum 來獲取數組的總和,從而避免了代碼重復。

簡化方法

隨著時間的推移,隨著維護代碼的人越來越多,代碼庫容易變得陳舊和混亂。為保證代碼的清晰度和易維護性,就非常有必要對代碼進行重構,使其更易理解、維護和擴展。

在簡化方法的過程中,首先要識別出那些包含復雜嵌套邏輯和承擔過多職責的方法。接著,可以通過以下幾步來簡化它們:

  • 遵循單一責任原則(SRP)來調整方法的功能劃分。
  • 將部分功能提取出來,創建新的子方法。
  • 刪除無用和冗余的代碼。
  • 減少方法內部的嵌套層次,使其結構更清晰。

接下來,我們將通過一個 Java 代碼重構示例來具體展示如何簡化方法。

簡化前

public class ShoppingCart {
    private List<Item> items;

    // 計算總價
    public double calculateTotalCost() {
        double total = 0;
        for (Item item : items) {
            if (item.isDiscounted()) {
                total += item.getPrice() * 0.8;
            } else {
                total += item.getPrice();
            }
        }
        if (total > 100) {
            total -= 10;
        }
        return total;
    }
}

我們可以通過提取 calculateItemPrice 邏輯到 calculateItemPrice 和 applyDiscount 方法中,并使用三元運算符來簡化條件判斷,使上述示例更為簡潔。

簡化后

public class ShoppingCart {
    private List<Item> items;

    // 計算總價
    public double calculateTotalCost() {
        double total = 0;
        for (Item item : items) {
            total += calculateItemPrice(item);
        }
        total -= applyDiscount(total);
        return total;
    }

    // 計算價格
    private double calculateItemPrice(Item item) {
        return item.isDiscounted() ? item.getPrice() * 0.8 : item.getPrice();
    }

    // 獲取滿減
    private double applyDiscount(double total) {
        return total > 100 ? 10 : 0;
    }
}

紅綠重構流程

圖片來源: Codecademy圖片來源: Codecademy

紅綠重構,又稱測試驅動開發(TDD),是一種強調先編寫測試再編寫能通過這些測試的代碼的代碼重構技術。該技術是一個循環迭代的過程,每一輪迭代都包括編寫新的測試和足夠的代碼來通過這些測試,最后對代碼進行重構。

這一技術包括以下三個階段:

  • 紅色階段:  在這一階段,你還未編寫實際的代碼。首先需要編寫一組預期會失敗的測試(標記為紅色),因為還沒有相應的實現來滿足這些測試條件。
  • 綠色階段:  此階段的目的是編寫足夠的代碼來通過之前編寫的未通過的測試,即讓測試變綠。注意,此時的目標不是編寫完美或高度優化的代碼,而是簡單地確保測試的通過。
  • 重構階段: 在確認代碼已成功通過所有測試后,此時應著重于代碼重構,以提升其性能和結構,而不改變其基本功能,確保測試仍然能夠順利通過。

每完成一個測試用例后,你將進入下一個循環,繼續編寫新的測試用例和對應的代碼,然后再進行代碼重構以實現更好的優化。

優化違反單一責任原則的代碼

可能你已對面向對象編程中的 SOLID 原則有所了解。SOLID 是五個設計原則的首字母縮寫。

  • S:單一責任原則(Single Responsibility Principle),這個原則強調一個類應該僅有一個變化的原因,簡言之,一個類應只負責一個功能點。
  • O:開放封閉原則(Open Closed Principle),軟件實體應該是可擴展的而不是可修改的。
  • L:里氏替換原則(Liskov Substitution Principle),對象應該能夠被其子類型所替換,而不影響程序的正確性。
  • I:接口隔離原則(Interface Segregation Principle),客戶端不應被強迫依賴于它們不用的接口。
  • D:依賴倒置原則(Dependency Inversion Principle),高層模塊不應該依賴于低層模塊,二者都應該依賴于抽象。

單一責任原則是首要原則,該原則強調每個類應僅有一個變化的原因,即一個類只負責一個功能點遵守單一責任原則是確保代碼可維護、可讀、靈活和模塊化的基本方式之一。下面我們將展示一個 OrderProcessor 類的示例,這個類違反了單一責任原則,因為它同時承擔了訂單處理和記錄信息日志兩項職責。

重構前

public class OrderProcessor {

    // 處理訂單
    public void processOrder(Order order) {
        // 訂單驗證

        
        // 處理邏輯
        
        // 日志記錄
        Logger logger = new Logger();
        logger.log("Order processed: " + order.getId());
    }
}

為了遵循單一責任原則,我們可以將該類重構為三個類:一個是 OrderProcessor 類,只負責訂單處理;OrderValidator負責訂單校驗,另外一個是 OrderLogger 類,專門負責日志記錄。

重構后

public class OrderProcessor {
    
    private final OrderValidator orderValidator;
    
    public OrderProcessor(OrderValidator orderValidator) {
        this.orderValidator = orderValidator;
    }
    
    // 處理訂單
    public void processOrder(Order order) {
        // 訂單驗證
        if(!orderValidator.validate(order)) {
            throw new IllegalArgumentException("Order is not valid");
        }
        
        // 處理邏輯
        // ...

        // 日志記錄
        OrderLogger logger = new OrderLogger();
        logger.logOrderProcessed(order);
    }
}

public class OrderValidator {

    // 訂單驗證
    public boolean validate(Order order) {
        // 驗證邏輯
        // ...
        return true;
    }
}


public class OrderLogger {

    
    public void logOrderProcessed(Order order) {
        Logger logger = new Logger();
        logger.log("Order processed: " + order.getId());
    }
}

Java 代碼重構的 14 條最佳實踐

代碼重構是一個能夠顯著提升代碼質量的重要步驟,它帶來了我們之前強調的諸多好處。但在進行重構時也需謹慎,特別是在處理龐大的代碼庫或不熟悉的代碼庫時,以避免無意中改變軟件的功能或產生未預見的問題。

為了避免任何潛在問題,您可以參考以下重構 Java 代碼的技巧和最佳實踐:

  1. 保持功能不變: Java 代碼重構的首要目標是提高代碼質量。然而,程序的外部行為,如它如何響應輸入和輸出以及其他用戶交互應該保持不變。
  2. 充分理解代碼: 在開始重構某個代碼片段之前,請確保你充分理解你即將重構的代碼。這包括其功能、交互和依賴關系。這種理解將指導你的決策,并幫助你避免做出可能影響代碼功能的更改。
  3. 將 Java 重構過程分解為小步驟: 重構大型軟件,尤其是當你對它還不夠了解時,可能會令人感到不知所措。然而,通過將重構過程分解為更小、易于管理的步驟,你可以使工作負擔更輕,減少錯誤的風險,并容易持續驗證你的更改。
  4. 使用版本控制創建備份: 由于重構涉及對代碼庫進行更改,有可能事情不會按計劃進行。在一個單獨的分支上備份你的工作軟件是一個好主意,以避免在找不出是哪些更改破壞了你的軟件時浪費大量時間和資源。像 Git 這樣的版本控制系統允許你同時管理不同的軟件版本。
  5. 頻繁測試你的更改: 在重構代碼時你最不想做的就是不小心破壞程序的功能或引入影響軟件的錯誤。在進行任何重大的代碼更改之前,尤其是重構,建立一套測試集是提供安全網的好方法。這些測試驗證你的代碼行為是否符合預期,以及現有的功能是否保持完整。
  6. 利用重構工具: 有了像 Eclipse 和 IntelliJ 這樣的現代 IDE,Java 代碼重構不必是一個緊張的過程。例如,IntelliJ IDEA 包括一套強大的重構功能。一些功能包括安全刪除,提取方法/參數/字段/變量,內聯重構,復制/移動等。這些功能使你的工作更輕松,減少了你在重構過程中引入錯誤的機會。
  7. 深入理解代碼變更:在重構過程中,深入了解你所做的代碼變更是非常重要的,它可以幫助你快速識別和解決可能出現的問題。
  8. 充分利用單元測試:作為開發者,我們需要確保在重構代碼時不會破壞現有的應用程序或引入新的 bug。一個完善的單元測試套件可以幫助你檢測回歸問題,確保功能的完整性,同時也有利于團隊合作和長期的代碼維護。
  9. 持續跟蹤性能變化并反饋:Java 代碼重構不僅僅是為了改善代碼結構,它還涉及到性能優化。通過持續監控性能指標,你可以確保你的重構工作正在取得實質性的進展。
  10. 進行同行代碼審查:重構完成后,建議邀請另一名開發者來審查你的更改,他們可以從一個新的角度來發現可能被你忽視的問題,并提供有價值的反饋。
  11. 記錄重構變更:當你與其他開發者一起工作時,記錄你的重構變更非常重要,因為這樣可以提高透明度和協作效率,同時也有助于新員工更快地了解項目。
  12. 進行回歸測試:完成重構后,需要通過回歸測試來確保所有現有的功能都得到了保留,并且新引入的邏輯沒有與現有代碼產生沖突。
  13. 保持團隊同步:在多人開發團隊中工作時,及時通報你的重構變更非常必要,這可以避免沖突并幫助團隊更好地適應新的變更。
  14. 在必要時執行回滾:如果重構過程中遇到無法解決的問題,不要猶豫,立即回滾到一個穩定的狀態,以避免浪費更多的時間和資源。

結論

重構是一項至關重要的技術實踐,它是確保軟件項目長期成功的關鍵。 通過融入我們之前討論的技術到你的開發周期并嚴格遵循最佳實踐,你可以把任何復雜且混亂的代碼庫改造為一個可讀、可維護和可擴展的軟件解決方案。但請注意,Java 代碼重構不是一次性的任務,而是一個可以持續整合到你的開發周期中的過程。

常見問題解答

應該何時重構我的 Java 代碼?

在軟件開發的任何階段都可以進行代碼重構。無論是在添加新功能、修復 bug 還是優化難以理解的代碼片段時,都是進行重構的好時機。定期預留時間來進行重構可以避免技術債務的累積,從而維持代碼庫的高質量。

如何決定哪些代碼需要重構?

你可以從識別代碼庫中難以理解、存在重復邏輯或容易產生 bug 的區域開始。尋找具有冗長的方法、復雜條件語句的代碼,并嘗試遵循“單一職責原則”來提高代碼的組織性。

如何確保重構不會引入 bug?

擁有一套完善的自動化測試套件是減少重構過程中引入 bug 的風險的關鍵。在開始重構之前,確保你的代碼具有良好的測試覆蓋率,這將幫助你捕捉任何可能的回歸問題,確保代碼功能的穩定性。

如何在 Java 中重構類?

首先,你需要確定目標類并深入理解其行為和依賴關系。然后可以考慮拆分龐大的方法和將方法移動到更適合的類中,同時利用繼承和接口來實現更清晰的結構。此外,重命名變量和方法、重新組織代碼結構和簡化條件語句都可以提高代碼的可讀性。最后,確保對所有更改進行徹底測試,以保證功能的完整性。

譯者介紹

劉汪洋,51CTO社區編輯,昵稱:明明如月,一個擁有 5 年開發經驗的某大廠高級 Java 工程師,擁有多個主流技術博客平臺博客專家稱號。

標題:CODE REFACTORING IN JAVA: TIPS, BEST PRACTICES, TECHNIQUES,作者:Digma

責任編輯:華軒 來源: 51CTO
相關推薦

2021-05-26 08:50:37

JavaScript代碼重構函數

2020-05-25 11:14:59

代碼程序開發

2024-08-15 08:11:10

2020-03-09 14:10:48

代碼開發工具

2022-11-28 23:48:06

JavaScript編程語言技巧

2014-07-29 13:55:10

程序員代碼

2025-01-21 08:10:00

2021-09-07 09:00:00

開發測試工具

2023-03-27 15:05:10

Python技巧

2021-12-04 23:10:02

Java代碼開發

2024-08-06 12:35:42

C#代碼重構

2021-07-01 15:17:14

MYSQL存儲數據庫

2019-12-03 09:32:32

JavaScript代碼開發

2019-05-16 09:00:06

云原生監控日志管理

2020-09-27 18:29:03

代碼機器學習數據科學

2020-09-27 17:17:26

機器學習技術人工智能

2024-11-28 09:43:04

2023-10-10 10:57:12

JavaScript代碼優化

2023-04-06 00:15:03

JavaReentrantL線程

2019-01-16 09:00:00

DevOps性能測試軟件
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 久草中文在线 | 久久久www | 日本成人中文字幕 | 亚洲国产一区二区在线 | 高清国产午夜精品久久久久久 | 澳门永久av免费网站 | 国产免费一区 | 国产在线播放av | 99免费视频 | 成av人电影在线 | 久操伊人 | 色就是色欧美 | 中文字幕在线视频免费观看 | 亚洲一区二区在线免费观看 | 国产精品一二三区在线观看 | 色婷婷综合久久久中字幕精品久久 | 精品在线观看一区二区 | 欧美中文字幕一区二区三区亚洲 | 国产成人精品久久二区二区91 | 免费黄色在线观看 | 黑人巨大精品欧美一区二区免费 | 欧美偷偷操 | h视频免费在线观看 | 亚洲国产情侣自拍 | 91精品久久久久久久久久入口 | 在线一区观看 | 国产美女精品 | 日韩精品一区在线 | 亚洲精品一区二区三区在线 | 一级片网址 | 久久免费小视频 | 久草资源在线视频 | 欧美一区二区三区视频在线播放 | 日韩精品一区二区三区久久 | 亚洲视频免费在线观看 | 91av视频在线| 国产免费一区二区三区最新6 | 一区在线视频 | 欧美激情国产日韩精品一区18 | 色橹橹欧美在线观看视频高清 | 欧美天堂 |