溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

怎么把if-else重構成高質量代碼

發(fā)布時間:2021-10-21 17:37:43 來源:億速云 閱讀:144 作者:iii 欄目:編程語言

本篇內容介紹了“怎么把if-else重構成高質量代碼”的有關知識,在實際案例的操作過程中,不少人都會遇到這樣的困境,接下來就讓小編帶領大家學習一下如何處理這些情況吧!希望大家仔細閱讀,能夠學有所成!

為什么我們寫的代碼都是 if-else?

程序員想必都經(jīng)歷過這樣的場景:剛開始自己寫的代碼很簡潔,邏輯清晰,函數(shù)精簡,沒有一個 if-else,可隨著代碼邏輯不斷完善和業(yè)務的瞬息萬變:比如需要對入?yún)⑦M行類型和值進行判斷;這里要判斷下對象是否為 null;不同類型執(zhí)行不同的流程。

落地到具體實現(xiàn)只能不停地加 if-else 來處理,漸漸地,代碼變得越來越龐大,函數(shù)越來越長,文件行數(shù)也迅速突破上千行,維護難度也越來越大,到后期基本達到一種難以維護的狀態(tài)。

雖然我們都很不情愿寫出滿屏 if-else 的代碼,可邏輯上就是需要特殊判斷,很絕望,可也沒辦法避免啊。

其實回頭看看自己的代碼,寫 if-else 不外乎兩種場景:異常邏輯處理和不同狀態(tài)處理。

兩者最主要的區(qū)別是:異常邏輯處理說明只能一個分支是正常流程,而不同狀態(tài)處理都所有分支都是正常流程。搜索程序員白楠楠公眾號,送你一份Java面試題寶典

怎么理解?舉個例子:

 //舉例一:異常邏輯處理例子
 Object obj = getObj();
 if (obj != null) {
     //do something
 }else{
     //do something
 }
 
 //舉例二:狀態(tài)處理例子
Object obj = getObj();
if (obj.getType == 1) {
    //do something
}else if (obj.getType == 2) {
    //do something
}else{
   //do something
}

第一個例子 if (obj != null) 是異常處理,是代碼健壯性判斷,只有 if 里面才是正常的處理流程,else 分支是出錯處理流程;而第二個例子不管 type 等于 1,2 還是其他情況,都屬于業(yè)務的正常流程。對于這兩種情況重構的方法也不一樣。

代碼 if-else 代碼太多有什么缺點?

缺點相當明顯了:最大的問題是代碼邏輯復雜,維護性差,極容易引發(fā) bug。如果使用 if-else,說明 if 分支和 else 分支的重視是同等的,但大多數(shù)情況并非如此,容易引起誤解和理解困難。

是否有好的方法優(yōu)化?如何重構?

方法肯定是有的。重構 if-else 時,心中無時無刻把握一個原則:

盡可能地維持正常流程代碼在最外層。

意思是說,可以寫 if-else 語句時一定要盡量保持主干代碼是正常流程,避免嵌套過深。

實現(xiàn)的手段有:減少嵌套、移除臨時變量、條件取反判斷、合并條件表達式等。

下面舉幾個實例來講解這些重構方法:

異常邏輯處理型重構方法實例一

重構前:

 double disablityAmount(){
     if(_seniority < 2)
         return 0;
 
     if(_monthsDisabled > 12)
         return 0;
 
     if(_isPartTime)
         return 0;

    //do somethig
}

重構后:

double disablityAmount(){
    if(_seniority < 2 || _monthsDisabled > 12 || _isPartTime)
        return 0;

    //do somethig
}

這里的重構手法叫合并條件表達式:如果有一系列條件測試都得到相同結果,將這些結果測試合并為一個條件表達式。

這個重構手法簡單易懂,帶來的效果也非常明顯,能有效地較少if語句,減少代碼量邏輯上也更加易懂。

異常邏輯處理型重構方法實例二

重構前:

 double getPayAmount(){
     double result;
     if(_isDead) {
         result = deadAmount();
     }else{
         if(_isSeparated){
             result = separatedAmount();
        }
         else{
            if(_isRetired){
                result = retiredAmount();
            else{
                result = normalPayAmount();
            }
        }
    }
    return result;
}

重構后:

 double getPayAmount(){
     if(_isDead)
         return deadAmount();
 
     if(_isSeparated)
        return separatedAmount();
 
     if(_isRetired)
         return retiredAmount();

    return normalPayAmount();
}

怎么樣?比對兩個版本,會發(fā)現(xiàn)重構后的版本邏輯清晰,簡潔易懂。

和重構前到底有什么區(qū)別呢?

最大的區(qū)別是減少 if-else 嵌套??梢钥吹?,最初的版本 if-else 最深的嵌套有三層,看上去邏輯分支非常多,進到里面基本都要被繞暈。其實,仔細想想嵌套內的 if-else 和最外層并沒有關聯(lián)性的,完全可以提取最頂層。

改為平行關系,而非包含關系,if-else 數(shù)量沒有變化,但是邏輯清晰明了,一目了然。

另一個重構點是廢除了 result 臨時變量,直接 return 返回。好處也顯而易見直接結束流程,縮短異常分支流程。原來的做法先賦值給 result 最后統(tǒng)一 return,那么對于最后 return 的值到底是那個函數(shù)返回的結果不明確,增加了一層理解難度。

總結重構的要點:如果 if-else 嵌套沒有關聯(lián)性,直接提取到第一層,一定要避免邏輯嵌套太深。盡量減少臨時變量改用 return 直接返回。

異常邏輯處理型重構方法實例三

重構前:

public double getAdjustedCapital(){
    double result = 0.0;
    if(_capital > 0.0 ){
        if(_intRate > 0 && _duration >0){
            resutl = (_income / _duration) *ADJ_FACTOR;
        }
    }
    return result;
}

第一步,運用第一招,減少嵌套和移除臨時變量:

public double getAdjustedCapital(){
    if(_capital <= 0.0 ){
        return 0.0;
    }
    if(_intRate > 0 && _duration >0){
        return (_income / _duration) *ADJ_FACTOR;
    }
    return 0.0;
}

這樣重構后,還不夠,因為主要的語句 (_income / _duration) *ADJ_FACTOR; 在 if 內部,并非在最外層,根據(jù)優(yōu)化原則(盡可能地維持正常流程代碼在最外層),可以再繼續(xù)重構:

 public double getAdjustedCapital(){
     if(_capital <= 0.0 ){
         return 0.0;
     }
     if(_intRate <= 0 || _duration <= 0){
         return 0.0;
     }
 
     return (_income / _duration) *ADJ_FACTOR;
}

這才是好的代碼風格,邏輯清晰,一目了然,沒有 if-else 嵌套難以理解的流程。

這里用到的重構方法是:將條件反轉使異常情況先退出,讓正常流程維持在主干流程。

異常邏輯處理型重構方法實例四

重構前:

    /* 查找年齡大于18歲且為男性的學生列表 */
     public ArrayList<Student> getStudents(int uid){
         ArrayList<Student> result = new ArrayList<Student>();
         Student stu = getStudentByUid(uid);
         if (stu != null) {
             Teacher teacher = stu.getTeacher();
             if(teacher != null){
                 ArrayList<Student> students = teacher.getStudents();
                 if(students != null){
                    for(Student student : students){
                        if(student.getAge() > = 18 && student.getGender() == MALE){
                            result.add(student);
                        }
                    }
                }else {
                    logger.error("獲取學生列表失敗");
                }
            }else {
                logger.error("獲取老師信息失敗");
            }
        } else {
            logger.error("獲取學生信息失敗");
        }
        return result;
    }

典型的"箭頭型"代碼,最大的問題是嵌套過深,解決方法是異常條件先退出,保持主干流程是核心流程:

重構后:

    /* 查找年齡大于18歲且為男性的學生列表 */
     public ArrayList<Student> getStudents(int uid){
         ArrayList<Student> result = new ArrayList<Student>();
         Student stu = getStudentByUid(uid);
         if (stu == null) {
             logger.error("獲取學生信息失敗");
             return result;
         }
 
        Teacher teacher = stu.getTeacher();
        if(teacher == null){
            logger.error("獲取老師信息失敗");
            return result;
        }

        ArrayList<Student> students = teacher.getStudents();
        if(students == null){
            logger.error("獲取學生列表失敗");
            return result;
        }

        for(Student student : students){
            if(student.getAge() > 18 && student.getGender() == MALE){
                result.add(student);
            }
        }
        return result;
    }

狀態(tài)處理型重構方法實例一

重構前:

 double getPayAmount(){
     Object obj = getObj();
     double money = 0;
     if (obj.getType == 1) {
         ObjectA objA = obj.getObjectA();
         money = objA.getMoney()*obj.getNormalMoneryA();
    }
     else if (obj.getType == 2) {
         ObjectB objB = obj.getObjectB();
        money = objB.getMoney()*obj.getNormalMoneryB()+1000;
    }
}

重構后:

 double getPayAmount(){
     Object obj = getObj();
     if (obj.getType == 1) {
         return getType1Money(obj);
     }
     else if (obj.getType == 2) {
         return getType2Money(obj);
     }
 }

double getType1Money(Object obj){
    ObjectA objA = obj.getObjectA();
    return objA.getMoney()*obj.getNormalMoneryA();
}

double getType2Money(Object obj){
    ObjectB objB = obj.getObjectB();
    return objB.getMoney()*obj.getNormalMoneryB()+1000;
}

這里使用的重構方法是:把 if-else 內的代碼都封裝成一個公共函數(shù)。函數(shù)的好處是屏蔽內部實現(xiàn),縮短 if-else 分支的代碼。代碼結構和邏輯上清晰,能一下看出來每一個條件內做的功能。

狀態(tài)處理型重構方法實例二

針對狀態(tài)處理的代碼,一種優(yōu)雅的做法是用多態(tài)取代條件表達式(《重構》推薦做法)。

你手上有個條件表達式,它根據(jù)對象類型的不同而選擇不同的行為。將這個表達式的每個分支放進一個子類內的覆寫函數(shù)中,然后將原始函數(shù)聲明為抽象函數(shù)。

重構前:

 double getSpeed(){
     switch(_type){
         case EUROPEAN:
            return getBaseSpeed();
         case AFRICAN:
             return getBaseSpeed()-getLoadFactor()*_numberOfCoconuts;
         case NORWEGIAN_BLUE:
             return (_isNailed)?0:getBaseSpeed(_voltage);
     }
}

重構后:

 class Bird{
     abstract double getSpeed();
 }
 
 class European extends Bird{
     double getSpeed(){
         return getBaseSpeed();
    }
 }

class African extends Bird{
    double getSpeed(){
        return getBaseSpeed()-getLoadFactor()*_numberOfCoconuts;
    }
}

class NorwegianBlue extends Bird{
   double getSpeed(){
        return (_isNailed)?0:getBaseSpeed(_voltage);
    }
}

可以看到,使用多態(tài)后直接沒有了 if-else,但使用多態(tài)對原來代碼修改過大,需要一番功夫才行。最好在設計之初就使用多態(tài)方式。

總結

if-else 代碼是每一個程序員最容易寫出的代碼,同時也是最容易被寫爛的代碼,稍不注意,就產(chǎn)生一堆難以維護和邏輯混亂的代碼。

針對條件型代碼重構把握一個原則:

盡可能地維持正常流程代碼在最外層,保持主干流程是正常核心流程。

為維持這個原則:合并條件表達式可以有效地減少if語句數(shù)目;減少嵌套能減少深層次邏輯;異常條件先退出自然而然主干流程就是正常流程。

針對狀態(tài)處理型重構方法有兩種:一種是把不同狀態(tài)的操作封裝成函數(shù),簡短 if-else 內代碼行數(shù);另一種是利用面向對象多態(tài)特性直接干掉了條件判斷。

現(xiàn)在回頭看看自己的代碼,犯了哪些典型錯誤,趕緊運用這些重構方法重構代碼吧?。?/p>

歡迎關注公眾號:程序員白楠楠,領取一份300頁pdf文檔的Java核心知識點總結!

這些資料的內容都是面試時面試官必問的知識點,篇章包括了很多知識點,其中包括了有基礎知識、Java集合、JVM、多線程并發(fā)、spring原理、微服務、Netty 與RPC 、Kafka、日記、設計模式、Java算法、數(shù)據(jù)庫、Zookeeper、分布式緩存、數(shù)據(jù)結構等等。

“怎么把if-else重構成高質量代碼”的內容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關的知識可以關注億速云網(wǎng)站,小編將為大家輸出更多高質量的實用文章!

向AI問一下細節(jié)

免責聲明:本站發(fā)布的內容(圖片、視頻和文字)以原創(chuàng)、轉載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權內容。

AI