当前位置:首页 > 公众号精选 > 21ic电子网
[导读]今天分享的文章不仅仅介绍9种应对代码中如果if-else比较多,阅读起来比较困难,维护起来也比较麻烦,也容易出Bug的情况,但是有些人也会有不同的看法!正文如下:观点一(灵剑):前期迭代懒得优化,来一个需求,加一个if,久而久之,就串成了一座金字塔。当代码已经复杂到难以维护的程度...

今天分享的文章不仅仅介绍9种应对代码中如果 if-else 比较多,阅读起来比较困难,维护起来也比较麻烦,也容易出 Bug的情况,但是有些人也会有不同的看法!
正文如下:

观点一(灵剑):

前期迭代懒得优化,来一个需求,加一个if,久而久之,就串成了一座金字塔。
当代码已经复杂到难以维护的程度之后,只能狠下心重构优化。那,有什么方案可以优雅的优化掉这些多余的if/else?

1. 提前return

这是判断条件取反的做法,代码在逻辑表达上会更清晰,看下面代码:if (condition) {
 // do something
else {
  return xxx;
}
其实,每次看到上面这种代码,我都心里抓痒,完全可以先判断!condition,干掉else。if (!condition) {
  return xxx;


// do something

2. 策略模式

有这么一种场景,根据不同的参数走不同的逻辑,其实这种场景很常见。
最一般的实现:
if (strategy.equals("fast")) {
  // 快速执行
else if (strategy.equals("normal")) {
  // 正常执行
else if (strategy.equals("smooth")) {
  // 平滑执行
else if (strategy.equals("slow")) {
  // 慢慢执行
}
看上面代码,有4种策略,有两种优化方案。

2.1 多态

interface Strategy {
  void run() throws Exception;
}

class FastStrategy implements Strategy {
    @Override
    void run() throws Exception {
        // 快速执行逻辑
    }
}

class NormalStrategy implements Strategy {
    @Override
    void run() throws Exception {
        // 正常执行逻辑
    }
}

class SmoothStrategy implements Strategy {
    @Override
    void run() throws Exception {
        // 平滑执行逻辑
    }
}

class SlowStrategy implements Strategy {
    @Override
    void run() throws Exception {
        // 慢速执行逻辑
    }
}
具体策略对象存放在一个Map中,优化后的实现Strategy strategy = map.get(param);
strategy.run();
上面这种优化方案有一个弊端,为了能够快速拿到对应的策略实现,需要map对象来保存策略,当添加一个新策略的时候,还需要手动添加到map中,容易被忽略。

2.2 枚举

发现很多同学不知道在枚举中可以定义方法,这里定义一个表示状态的枚举,另外可以实现一个run方法。public enum Status {
    NEW(0) {
      @Override
      void run() {
        //do something  
      }
    },
    RUNNABLE(1) {
      @Override
       void run() {
         //do something  
      }
    };

    public int statusCode;

    abstract void run();

    Status(int statusCode){
        this.statusCode = statusCode;
    }
}
重新定义策略枚举public enum Strategy {
    FAST {
      @Override
      void run() {
        //do something  
      }
    },
    NORMAL {
      @Override
       void run() {
         //do something  
      }
    },

    SMOOTH {
      @Override
       void run() {
         //do something  
      }
    },

    SLOW {
      @Override
       void run() {
         //do something  
      }
    };
    abstract void run();
}
通过枚举优化之后的代码如下Strategy strategy = Strategy.valueOf(param);
strategy.run();

3. 学会使用 Optional

Optional主要用于非空判断,由于是jdk8新特性,所以使用的不是特别多,但是用起来真的爽。使用之前:if (user == null) {
    //do action 1
else {
    //do action2
}
如果登录用户为空,执行action1,否则执行action 2,使用Optional优化之后,让非空校验更加优雅,间接的减少if操作Optional userOptional = Optional.ofNullable(user);
userOptional.map(action1).orElse(action2);

4. 数组小技巧

来自google解释,这是一种编程模式,叫做表驱动法,本质是从表里查询信息来代替逻辑语句,比如有这么一个场景,通过月份来获取当月的天数,仅作为案例演示,数据并不严谨。一般的实现:int getDays(int month){
    if (month == 1)  return 31;
    if (month == 2)  return 29;
    if (month == 3)  return 31;
    if (month == 4)  return 30;
    if (month == 5)  return 31;
    if (month == 6)  return 30;
    if (month == 7)  return 31;
    if (month == 8)  return 31;
    if (month == 9)  return 30;
    if (month == 10)  return 31;
    if (month == 11)  return 30;
    if (month == 12)  return 31;
}
优化后的代码int monthDays[12] = {312931303130313130313031};
int getDays(int month){
    return monthDays[--month];
}

结束

if else作为每种编程语言都不可或缺的条件语句,在编程时会大量的用到。一般建议嵌套不要超过三层,如果一段代码存在过多的if else嵌套,代码的可读性就会急速下降,后期维护难度也大大提高。

观点二(IT技术控):

不要去过度关注if/else的层数,而要关注接口语义是否足够清晰;单纯减少if/else的层数,然后拆出一堆do_logic1, do_logic2…这样的接口是毫无帮助的。任何一个接口的执行过程都可以表示为:输入 内部状态 -> 输出这样的形式,我们分以下几种情况来讨论:输入、内部状态、输出都很简单,但中间逻辑复杂。比如说一个精心优化过的数值计算程序,可能需要根据输入在不同的取值范围采取不同的策略,还有很多逻辑用来处理会引发问题(比如除0)的边界值,这种情况下if/else数量多是难以避免的,根据步骤拆分出一些内部方法有一定帮助,但也不能完全解决问题。这种情况下最好的做法是写一篇详细的文档,从最原始的数学模型开始,然后表明什么情况下采取什么样的计算策略,策略如何推导,知道得到代码中使用的具体形式,然后给整个方法加上注释附上文档地址,并且在每个分支的地方加上注释指明对应到文档中哪个公式。这种情况下虽然方法很复杂,但是语义是清晰的,如果不修改实现的话理解语义就行了,如果要修改实现那么需要参考对照文档中的公式。输入过于复杂,比如输入带有一堆不同的参数,或者有各种奇怪的flag,每个flag有不同作用。这种情况下首先需要提高接口的抽象层次:如果接口有多个不同作用,需要拆分成不同接口;如果接口内部根据不同参数进不同分支,需要将这些参数和对应分支包在Adapter里,使用参数的地方改写成Adapter的接口,根据传入的Adapter类型不同进入不同的实现;如果接口内部有复杂的参数转换关系,需要改写成查找表。这种情况下的主要问题是接口本身抽象的有问题,有更清晰的抽象之后,实现也自然没有那么多if/else了。输出过于复杂,为了省事一个过程计算出了太多东西,又为了性能加了一堆flag控制是否计算之类。这种情况下需要果断将方法拆分成多个不同方法,每个方法只返回自己需要的内容。如果不同计算之间有共用的内部结果呢?如果这个内部结果计算并不形成瓶颈,只要提取出内部方法然后在不同过程中分别调用即可;如果希望避免重复计算,可以增加一个额外的cache对象作为参数,cache内容对用户不透明,用户只保证相同输入使用同一个cache对象即可,在计算中将中间结果保存到cache中,下次计算前先检查有没有已经得到的结果,就可以避免重复计算了。内部状态过于复杂。首先检查状态设置的是否合理,是不是有一些本来应该作为输入参数的东西被放到了内部状态中(比如用来隐式地在两个不同方法调用之间传递参数)?其次,这些状态分别控制哪些方面,是否可以分组然后实现到不同的StateManager里面?第三,画出状态转移图,尝试将内部状态分成单层分支,然后分别实现到on_xxx_state这样的方法里面,然后通过单层的switch或者查找表来调用。其实通常需要优化的都是整体接口抽象,而不是单个接口的实现,单个接口实现不清晰通常是因为接口实现和需求不同构造成的。来源:https://www.zhihu.com/question/344856665版权归原作者所有,如有侵权,请联系删除。select_card mp_profile_iframe" data-pluginname="mpprofile" data-id="MjM5NDQ0NjM5Mg==" data-headimg="http://mmbiz.qpic.cn/mmbiz_png/XZMicpgcYibXGZlfHyXUq0wvbDaJ3ic0P6E1Gic49tLOl2sa8GcMFwIIpibpARTJRbwmHqKoYc9jcY5fr4e7dFjNMmA/0?wx_fmt=png" data-nickname="21ic电子网" data-alias="weixin21ic" data-signature="即时传播最新电子科技信息,汇聚业界精英精彩视点。" data-from="0">

21ic电子网

扫描二维码,关注更多精彩内容

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

持续引进行业高级人才 上海2022年4月18日 /美通社/ -- 近日,全球领先的自动驾驶智能环卫公司仙途智能autowise.ai 公布2022 Q1营收情况。数据显示,2022年Q1,autowise.a...

关键字: se

(全球TMT2022年4月18日讯)自动驾驶智能环卫公司仙途智能autowise.ai 公布2022 Q1营收情况。数据显示,2022年Q1,autowise.ai实现营收数千万元,在执行合同额达数亿元,创历史新高,实...

关键字: se

生于海,长于山 上海2022年4月2日 /美通社/ -- 深受世界户外爱好者及探险家青睐的挪威国宝级户外品牌海丽汉森Helly Hansen(简称HH),创立以来始终秉持专业精神,专注打造高端专业户外装备。近...

关键字: se rc

深圳新办事处2022年全速推进,快速招聘顶尖人才 苏黎世2022年3月30日 /美通社/ -- 全球性专业服务公司Synpulse今天宣布在数...

关键字: se npu

北京2022年3月29日 /美通社/ -- Analog Devices, Inc. (NASDAQ: ADI)和Gridspertise宣布携手合作提高全球智能电网的弹性和质量。Gridspertise是一家...

关键字: se 电气 智能电网

(全球TMT2022年3月29日讯)Analog Devices, Inc. 和Gridspertise宣布携手合作提高全球智能电网的弹性和质量。Gridspertise是一家为配电系统运营商(DSO)提供先...

关键字: ADI 智能电网 se

北京2022年3月24日 /美通社/ -- 3月22日,浪潮信息宣布元宇宙服务器MetaEngine全面支持NVIDIA Omniverse Enterprise,提供支撑大规模数字孪生场景的软硬件一体化解决方案OVX,...

关键字: 英伟达 se ni

(全球TMT2022年3月23日讯)中国电信股份有限公司公布,公司在《The Asset》的‘2021年ESG企业大奖’评选中,勇夺‘杰出环境、社会及管治企业金奖’。同时,执行董事、董事长兼首席执行官柯瑞文先生在此次评...

关键字: 中国电信 se

(全球TMT2022年3月2日讯)为全球领先品牌提供营销解决方案的全球性公司RTB House宣布,资深的数字广告专家Shien Zhu(祝师恩) 将加入其东京办事处,担任亚太区品牌销售副总裁。正值该公司继续...

关键字: se

新加坡2022年3月1日 /美通社/ -- 为全球领先品牌提供先进营销解决方案的全球性公司RTB House今天宣布,资深的数字广告专家Shien Zhu(祝师恩) ...

关键字: se
关闭
关闭