如何優(yōu)化if/else代碼

這篇文章主要介紹“如何優(yōu)化if/ else代碼”,在日常操作中,相信很多人在如何優(yōu)化if/ else代碼問題上存在疑惑,小編查閱了各式資料,整理出簡單好用的操作方法,希望對大家解答”如何優(yōu)化if/ else代碼”的疑惑有所幫助!接下來,請跟著小編一起來學習吧!

網站制作、成都網站建設,成都做網站公司-創(chuàng)新互聯已向超過千家企業(yè)提供了,網站設計,網站制作,網絡營銷等服務!設計與技術結合,多年網站推廣經驗,合理的價格為您打造企業(yè)品質網站。

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

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

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

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

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

兩者最主要的區(qū)別是:異常邏輯處理說明只能一個分支是正常流程,而不同狀態(tài)處理都所有分支都是正常流程。

怎么理解?舉個例子:

1//舉例一:異常邏輯處理例子
 2Object obj = getObj();
 3if (obj != null) {
 4    //do something
 5}else 6    //do something
 7}
 8 9//舉例二:狀態(tài)處理例子10Object obj = getObj();11if (obj.getType == 1) {12    //do something13}else if (obj.getType == 2) {14    //do something15}else{16    //do something17}

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

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

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

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

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

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

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

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

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

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

重構前:

1double disablityAmount(){
 2    if(_seniority < 2)
 3        return 0;
 4 5    if(_monthsDisabled > 12)
 6        return 0;
 7 8    if(_isPartTime)
 9        return 0;1011    //do somethig12}

重構后:

1double disablityAmount(){2    if(_seniority < 2 || _monthsDisabled > 12 || _isPartTime)3        return 0;45    //do somethig6}

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

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

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

重構前:

1double getPayAmount(){
 2    double result;
 3    if(_isDead) {
 4        result = deadAmount();
 5    }else{
 6        if(_isSeparated){
 7            result = separatedAmount();
 8        }
 9        else{10            if(_isRetired){11                result = retiredAmount();12            else{13                result = normalPayAmount();14            }15        }16    }17    return result;18}

重構后:

1double getPayAmount(){
 2    if(_isDead) 3        return deadAmount();
 4 5    if(_isSeparated) 6        return separatedAmount();
 7 8    if(_isRetired) 9        return retiredAmount();1011    return normalPayAmount();12}

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

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

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

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

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

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

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

重構前:

1public double getAdjustedCapital(){2    double result = 0.0;3    if(_capital > 0.0 ){4        if(_intRate > 0 && _duration >0){5            resutl = (_income / _duration) *ADJ_FACTOR;6        }7    }8    return result;9}

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

1public double getAdjustedCapital(){2    if(_capital <= 0.0 ){3        return 0.0;4    }5    if(_intRate > 0 && _duration >0){6        return (_income / _duration) *ADJ_FACTOR;7    }8    return 0.0;9}

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

1public double getAdjustedCapital(){
 2    if(_capital <= 0.0 ){
 3        return 0.0;
 4    }
 5    if(_intRate <= 0 || _duration <= 0){
 6        return 0.0;
 7    }
 8 9    return (_income / _duration) *ADJ_FACTOR;10}

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

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

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

重構前:

1   /* 查找年齡大于18歲且為男性的學生列表 */ 2    public ArrayList<Student> getStudents(int uid){
 3        ArrayList<Student> result = new ArrayList<Student>();
 4        Student stu = getStudentByUid(uid);
 5        if (stu != null) {
 6            Teacher teacher = stu.getTeacher();
 7            if(teacher != null){
 8                ArrayList<Student> students = teacher.getStudents();
 9                if(students != null){10                    for(Student student : students){11                        if(student.getAge() > = 18 && student.getGender() == MALE){12                            result.add(student);13                        }14                    }15                }else {16                    logger.error("獲取學生列表失敗");17                }18            }else {19                logger.error("獲取老師信息失敗");20            }21        } else {22            logger.error("獲取學生信息失敗");23        }24        return result;25    }

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

重構后:

1   /* 查找年齡大于18歲且為男性的學生列表 */ 2    public ArrayList<Student> getStudents(int uid){
 3        ArrayList<Student> result = new ArrayList<Student>();
 4        Student stu = getStudentByUid(uid);
 5        if (stu == null) {
 6            logger.error("獲取學生信息失敗");
 7            return result;
 8        }
 910        Teacher teacher = stu.getTeacher();11        if(teacher == null){12            logger.error("獲取老師信息失敗");13            return result;14        }1516        ArrayList<Student> students = teacher.getStudents();17        if(students == null){18            logger.error("獲取學生列表失敗");19            return result;20        }2122        for(Student student : students){23            if(student.getAge() > 18 && student.getGender() == MALE){24                result.add(student);25            }26        }27        return result;28    }
狀態(tài)處理型重構方法實例一

重構前:

1double getPayAmount(){
 2    Object obj = getObj();
 3    double money = 0;
 4    if (obj.getType == 1) {
 5        ObjectA objA = obj.getObjectA();
 6        money = objA.getMoney()*obj.getNormalMoneryA();
 7    }
 8    else if (obj.getType == 2) {
 9        ObjectB objB = obj.getObjectB();10        money = objB.getMoney()*obj.getNormalMoneryB()+1000;11    }12}

重構后:

1double getPayAmount(){
 2    Object obj = getObj();
 3    if (obj.getType == 1) {
 4        return getType1Money(obj);
 5    }
 6    else if (obj.getType == 2) {
 7        return getType2Money(obj);
 8    }
 9}1011double getType1Money(Object obj){12    ObjectA objA = obj.getObjectA();13    return objA.getMoney()*obj.getNormalMoneryA();14}1516double getType2Money(Object obj){17    ObjectB objB = obj.getObjectB();18    return objB.getMoney()*obj.getNormalMoneryB()+1000;19}

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

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

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

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

重構前:

1double getSpeed(){
 2    switch(_type){
 3        case EUROPEAN: 4            return getBaseSpeed();
 5        case AFRICAN: 6            return getBaseSpeed()-getLoadFactor()*_numberOfCoconuts;
 7        case NORWEGIAN_BLUE: 8            return (_isNailed)?0:getBaseSpeed(_voltage);
 9    }10}

重構后:

1class Bird{
 2    abstract double getSpeed();
 3}
 4 5class European extends Bird{
 6    double getSpeed(){
 7        return getBaseSpeed();
 8    }
 9}1011class African extends Bird{12    double getSpeed(){13        return getBaseSpeed()-getLoadFactor()*_numberOfCoconuts;14    }15}1617class NorwegianBlue extends Bird{18    double getSpeed(){19        return (_isNailed)?0:getBaseSpeed(_voltage);20    }21}

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

到此,關于“如何優(yōu)化if/ else代碼”的學習就結束了,希望能夠解決大家的疑惑。理論與實踐的搭配能更好的幫助大家學習,快去試試吧!若想繼續(xù)學習更多相關知識,請繼續(xù)關注創(chuàng)新互聯網站,小編會繼續(xù)努力為大家?guī)砀鄬嵱玫奈恼拢?/p>

新聞標題:如何優(yōu)化if/else代碼
轉載來源:http://muchs.cn/article6/jcphog.html

成都網站建設公司_創(chuàng)新互聯,為您提供網頁設計公司品牌網站建設、建站公司、營銷型網站建設微信公眾號域名注冊

廣告

聲明:本網站發(fā)布的內容(圖片、視頻和文字)以用戶投稿、用戶轉載內容為主,如果涉及侵權請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網站立場,如需處理請聯系客服。電話:028-86922220;郵箱:631063699@qq.com。內容未經允許不得轉載,或轉載時需注明來源: 創(chuàng)新互聯

搜索引擎優(yōu)化