爛慫if-else代碼優(yōu)化方案
0.問題概述
代碼可讀性是衡量代碼質(zhì)量的重要標(biāo)準(zhǔn),可讀性也是可維護(hù)性、可擴(kuò)展性的保證,因?yàn)榇a是連接程序員和機(jī)器的中間橋梁,要對(duì)雙邊友好。Quora 上有一個(gè)帖子: “What are some of the most basic things every programmer should know?”
其中:
Code that’s hard to understand is hard to maintain.
Code that’s hard to maintain is next to useless.
也強(qiáng)調(diào)了"easy understand"代碼的重要性。
寫這篇文章的契機(jī)是在研讀Apache ShenYu項(xiàng)目時(shí),看到了很大一坨的if else語句,如下:
這里并非評(píng)論這段代碼寫法有問題,因?yàn)槲疫€并沒有深入到項(xiàng)目細(xì)節(jié)之中,可能這已經(jīng)是多輪優(yōu)化的結(jié)果嘞。
但是這個(gè)多層if else的形式引發(fā)了我的思考,因?yàn)槲乙苍陧?xiàng)目代碼中引入過如此繁重的if else結(jié)構(gòu),并在Code Review中被指出了問題。從那以后,我對(duì)if else的最大容忍層數(shù)就是三層。
我把大量if else的場(chǎng)景按照深度和廣度兩個(gè)維度劃分為兩種情況:
嵌套層級(jí)過深
平鋪范圍太廣
下面就討論一下,當(dāng)代碼中存在大量這樣結(jié)構(gòu)的代碼的時(shí)候,該如何優(yōu)化?
1.解決方案
1.1 盡早返回
又稱衛(wèi)語句,即Guard Statement
WikiPedia:
In computer programming, aguardis abooleanexpressionthat must evaluate to true if the program execution is to continue in the branch in question.
Regardless of which programming language is used, aguard clause,guard code, orguard statement, is a check of integritypreconditionsused to avoid errors during execution. A typical example is checking that a reference about to be processed is not null, which avoids null-pointer failures. Other uses include using a boolean field foridempotence(so subsequent calls are nops), as in thedispose pattern. The guard provides anearly exitfrom asubroutine, and is a commonly used deviation fromstructured programming, removing one level of nesting and resulting in flatter code:[1]replacing
if guard { ... }
withif not guard: return; ...
.
實(shí)際應(yīng)用:
kotlin復(fù)制代碼if (CollectionUtils.isNotEmpty(list)) { ?// do something } else { ? ? return xxx; } ?
使用盡早返回優(yōu)化:
kotlin復(fù)制代碼if (CollectionUtils.isEmpty(list)) { ?return xxx; } ?// do something ?
可以看到,優(yōu)化后的代碼不僅節(jié)省了一個(gè)else語句,也能讓后續(xù)的"do something"節(jié)省一層if else包裹,代碼看起來更干凈一些
結(jié)合這個(gè)例子再說一下我對(duì)衛(wèi)語句的理解:
可以將“衛(wèi)”理解為“門衛(wèi)”,門衛(wèi)的作用是檢查過濾,只有符合條件的語句,才可以繼續(xù)執(zhí)行,否則直接勸返(return)。吐槽一下這種中文直譯有些晦澀,未免有點(diǎn)“德先生賽先生”的意思了。。。
1.2 使用switch或三元運(yùn)算符
可以利用語法知識(shí),對(duì)if else進(jìn)行簡化,
例如,當(dāng)if else滿足一定條件時(shí):
scss復(fù)制代碼if (condition1) { ?? ?doSomeThing1(); } else if (condition2) { ?? ?doSomeThing2(); } else if (condition3) { ?? ?doSomeThing3(); ?} else if (condition4) { ?? ?doSomeThing4(); } else { ?? ?doSomeThing5(); ?}... ?
可以使用switch case語法進(jìn)行替換
或,
例如使用三元運(yùn)算符進(jìn)行賦值操作:
Integer num = obejct == null ? 1 : object.value();
1.3 策略模式
1.3.1 概念
策略模式是一種行為設(shè)計(jì)模式,即一個(gè)對(duì)象有一個(gè)確定的行為,在不同場(chǎng)景下,這些行為有不同的算法實(shí)現(xiàn)。
例如從內(nèi)蒙通過公共交通去北京是一個(gè)確定的行為,在天上這種場(chǎng)景可以選擇飛機(jī),地上的場(chǎng)景可以選擇火車~
策略模式一般包含三個(gè)要素:
抽象策略(Abstract strategy):定義所謂的“確定的行為”,一般由接口或抽象類實(shí)現(xiàn)
具體實(shí)現(xiàn)(Concrete strategy):封裝對(duì)應(yīng)場(chǎng)景下的具體算法實(shí)現(xiàn)。
上下文(Context):負(fù)責(zé)具體實(shí)現(xiàn)策略的管理并供對(duì)象使用。
1.3.2 使用場(chǎng)景
一個(gè)接口或抽象類的各個(gè)子類都是為了解決相同的問題,區(qū)分這些子類的只有方法實(shí)現(xiàn)的不同。
代碼中使用大量if else或大面積switch case來選擇具體的子實(shí)現(xiàn)類
1.3.3 實(shí)際應(yīng)用
例如:
csharp復(fù)制代碼if ("man".equals(strategy)) { ? ? // Perform related operations ?} else if ("woman".equals(strategy)) { ? ? // Perform operations related to women } else if ("other".equals(strategy)) { ? ? // Perform other operations } ?
上面一段代碼,每一個(gè)if分支完成的都是相同的操作,只是在不同的性別場(chǎng)景下,操作方法的實(shí)現(xiàn)不同,那么就可以使用策略模式進(jìn)行優(yōu)化:
首先,定義一個(gè)抽象策略接口:
csharp復(fù)制代碼public interface Strategy { ? ? ?void run() throws Exception; ?} ?
然后,進(jìn)行不同策略的實(shí)現(xiàn):
java復(fù)制代碼//Men's strategy implementation class @Slf4j public class ManStrategy implements Strategy { ? ? ?@Override ?? ?public void run() throws Exception { ?? ? ? ?// Fast man's logic ?? ? ? ?log.debug("Execute the logic related to men..."); ?? ?} ?} ?//Women's strategy implementation class @Slf4j public class WomanStrategy implements Strategy { ? ? ?@Override ?? ?public void run() throws Exception { ?? ? ? ?// Fast woman's logic ?? ? ? ?log.debug("Execute women related logic..."); ?? ?} ?} ?//Others' policy implementation class @Slf4j public class OtherStrategy implements Strategy { ? ? ?@Override ?? ?public void run() throws Exception { ?? ? ? ?// Fast other logic ?? ? ? ?log.debug("Perform other related logic..."); ?? ?} ?} ?
最后,進(jìn)行策略的應(yīng)用:
arduino復(fù)制代碼public class StrategyTest { ? ? ?public static void main(String[] args) { ?? ? ? ?try { ?? ? ? ? ? ?Strategy strategy = initMap("man"); ?? ? ? ? ? ?strategy.run(); ?? ? ? ?} catch (Exception e) { ?? ? ? ? ? ?e.printStackTrace(); ?? ? ? ?} ?? ?} ? ? ?//Initialize the Map to obtain a gender policy ?? ?private static Strategy initMap(String key) { ?? ? ? ?//Use simple example ?? ? ? ?HashMap<String, Strategy> map = new HashMap<>(); ?? ? ? ?map.put("man", new ManStrategy()); ?? ? ? ?map.put("woman", new WomanStrategy()); ?? ? ? ?map.put("other", new OtherStrategy()); ?? ? ? ?return map.get(key); ?? ?} ?} ?
1.3.4 優(yōu)劣勢(shì)分析及優(yōu)化
1.3.4.1 劣勢(shì)
整體上來看,使用策略模式雖然剔除了大量的if else語句,但是也引入了更多的類文件,同時(shí)在Context中需要維護(hù)一個(gè)類似注冊(cè)表的map對(duì)象,當(dāng)增加策略實(shí)現(xiàn)時(shí),容易忘記。
優(yōu)化措施:
在Java中,可以使用函數(shù)式編程進(jìn)行優(yōu)化:
arduino復(fù)制代碼@Slf4j public class StrategyTest { ? ? ?public static void main(String[] args) { ?? ? ? ?//Use simple example ?? ? ? ?HashMap<String, Strategy> map = new HashMap<>(); ?? ? ? ?map.put("man", () -> log.debug("Execute the logic related to men...")); ?? ? ? ?map.put("woman", () -> log.debug("Execute women related logic...")); ?? ? ? ?map.put("other", () -> log.debug("Execute logic related to others...")); ? ? ? ? ?try { ?? ? ? ? ? ?map.get("woman").run(); ?? ? ? ?} catch (Exception e) { ?? ? ? ? ? ?e.printStackTrace(); ?? ? ? ?} ?? ?} } ?
或者,使用枚舉進(jìn)行優(yōu)化:
typescript復(fù)制代碼@Slf4j public enum Strategy { ? ? ?//Man state ?? ?MAN(0) { ?? ? ? ?@Override ?? ? ? ?void run() { ?? ? ? ? ? ?//Perform related operations ?? ? ? ? ? ?log.debug("Execute the logic related to men"); ?? ? ? ?} ?? ?}, ?? ?//Woman state ?? ?WOMAN(1) { ?? ? ? ?@Override ?? ? ? ?void run() { ?? ? ? ? ? ?//Perform operations related to women ?? ? ? ? ? ?log.debug("Execute women related logic"); ?? ? ? ?} ?? ?}, ?? ?//Other status ?? ?OTHER(2) { ?? ? ? ?@Override ?? ? ? ?void run() { ?? ? ? ? ? ?//Perform other related operations ?? ? ? ? ? ?log.debug("Perform other related logic"); ?? ? ? ?} ?? ?}; ? ? ?abstract void run(); ? ? ?public int statusCode; ? ? ?Strategy(int statusCode) { ?? ? ? ?this.statusCode = statusCode; ?? ?} ?} ?
typescript復(fù)制代碼public static void main(String[] args) { ?? ? ? ?try { ?? ? ? ? ? ?//Simple use example ?? ? ? ? ? ?String param = String.valueOf(Strategy.WOMAN); ?? ? ? ? ? ?Strategy strategy = Strategy.valueOf(param); ?? ? ? ? ? ?strategy.run(); ?? ? ? ?} catch (Exception e) { ?? ? ? ? ? ?e.printStackTrace(); ?? ? ? ?} } ?
除此以外,在客戶端實(shí)際使用策略時(shí),即對(duì)象進(jìn)行方法的調(diào)用時(shí),客戶端必須知道這個(gè)策略的所有實(shí)現(xiàn)子類,并需要了解這些子類之間的不同以及各自的應(yīng)用場(chǎng)景,這樣客戶端才能選擇合適的策略實(shí)現(xiàn)“確定的行為”。
1.3.4.2 優(yōu)勢(shì)
最直接的好處就是可以讓又臭又長的if else代碼塊看起來更干凈。
面向?qū)ο蟮娜筇攸c(diǎn):封裝、繼承、多態(tài),在策略模式中都能找到影子。面向接口編程,代碼的可擴(kuò)展性好
代碼的可測(cè)性好,Mock更方便,減少了分支判斷,實(shí)現(xiàn)類只需要各自測(cè)試即可。
1.4 Optional
if else分支判斷的很多情況都是進(jìn)行非空條件的判斷,Optional是Java8開始提供的新特性,使用這個(gè)語法特性,也可以減少代碼中if else的數(shù)量,例如:
優(yōu)化前:
rust復(fù)制代碼String str = "Hello World!"; ?if (str != null) { ?? ?System.out.println(str); } else { ?? ?System.out.println("Null"); } ?
優(yōu)化后:
vbnet復(fù)制代碼Optional<String> optional = Optional.of("Hello World!"); optional.ifPresentOrElse(System.out::println, () -> System.out.println("Null")); ?
1.5 注冊(cè)表
這種方式和策略模式有相似之處,但注冊(cè)表更自由,不需要提煉接口,只需要將自定義實(shí)現(xiàn)在注冊(cè)表中注冊(cè)即可。
例如,優(yōu)化前:
scss復(fù)制代碼if (param.equals(value1)) { ?? ?doAction1(someParams); }else if (param.equals(value2)) { ?? ?doAction2(someParams); }else if (param.equals(value3)) { ?? ?doAction3(someParams); } ?
優(yōu)化后:
scss復(fù)制代碼//Generic here? For the convenience of demonstration, it can be replaced with the real type we need in actual development Map<?, Function<?> action> actionMappings = new HashMap<>(); ?// When init actionMappings.put(value1, (someParams) -> { doAction1(someParams)}); actionMappings.put(value2, (someParams) -> { doAction2(someParams)}); actionMappings.put(value3, (someParams) -> { doAction3(someParams)}); ? // Omit null judgment actionMappings.get(param).apply(someParams); ?
1.6 責(zé)任鏈模式
先來看一段代碼:
vbscript復(fù)制代碼public void handle(request) { ?? ?if (handlerA.canHandle(request)) { ?? ? ? ?handlerA.handleRequest(request); ?? ?} else if (handlerB.canHandle(request)) { ?? ? ? ?handlerB.handleRequest(request); ?? ?} else if (handlerC.canHandle(request)) { ?? ? ? ?handlerC.handleRequest(request); ?? ?} } ?
代碼中也是存在一坨if else語句,但是和上述例子不同之處在于,if條件判斷權(quán)在每個(gè)handler組件中,每一個(gè)handler的判斷方式也可能不盡相同,相當(dāng)靈活,同一個(gè)request可能同時(shí)滿足多個(gè)if條件
解決方案就是參考開源組件中Filter或者Interceptor責(zé)任鏈機(jī)制,優(yōu)化后代碼:
typescript復(fù)制代碼public void handle(request) { ??handlerA.handleRequest(request); } ? public abstract class Handler { ? ??protected Handler next; ? ??public abstract void handleRequest(Request request); ? ??public void setNext(Handler next) { this.next = next; } } ? public class HandlerA extends Handler { ??public void handleRequest(Request request) { ?? ?if (canHandle(request)) doHandle(request); ?? ?else if (next != null) next.handleRequest(request); ??} } ?
2.總結(jié)&思考
這篇文章主要介紹了代碼中if else代碼塊泛濫時(shí)的治理措施,在實(shí)際應(yīng)用時(shí)可根據(jù)具體場(chǎng)景選擇合理的方案。
其實(shí)代碼中存在大面積if else本無問題,用一句網(wǎng)絡(luò)流行語來反駁就是:“你就說能不能用吧!”。但是作為有追求的工程師,我們要對(duì)項(xiàng)目以及代碼負(fù)責(zé),要及時(shí)的識(shí)別到代碼中的壞味道,并持續(xù)重構(gòu)優(yōu)化。最后還想說一定要擁抱開源,多研讀他人優(yōu)秀代碼,并臨摹、思考、實(shí)踐,日拱一卒,不期而至。