0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

有什么方案可以优雅的优化掉这些多余的if/else呢?

嵌入式开发爱好者 来源:嵌入式开发爱好者 2023-06-19 09:56 次阅读

观点一(灵剑):

前期迭代懒得优化,来一个需求,加一个if,久而久之,就串成了一座金字塔。

6dee380e-0df1-11ee-962d-dac502259ad0.png

当代码已经复杂到难以维护的程度之后,只能狠下心重构优化。那,有什么方案可以优雅的优化掉这些多余的if/else?

1. 提前 return

这是判断条件取反的做法,代码在逻辑表达上会更清晰,看下面代码:

if(condition){
//dosomething
}else{
returnxxx;
}

其实,每次看到上面这种代码,我都心里抓痒,完全可以先判断!condition,干掉 else。

if(!condition){
returnxxx;

}
//dosomething

2. 策略模式

有这么一种场景,根据不同的参数走不同的逻辑,其实这种场景很常见。最一般的实现:

if(strategy.equals("fast")){
//快速执行
}elseif(strategy.equals("normal")){
//正常执行
}elseif(strategy.equals("smooth")){
//平滑执行
}elseif(strategy.equals("slow")){
//慢慢执行
}

看上面代码,有4种策略,有两种优化方案。

2.1 多态

interfaceStrategy{
voidrun()throwsException;
}

classFastStrategyimplementsStrategy{
@Override
voidrun()throwsException{
//快速执行逻辑
}
}

classNormalStrategyimplementsStrategy{
@Override
voidrun()throwsException{
//正常执行逻辑
}
}

classSmoothStrategyimplementsStrategy{
@Override
voidrun()throwsException{
//平滑执行逻辑
}
}

classSlowStrategyimplementsStrategy{
@Override
voidrun()throwsException{
//慢速执行逻辑
}
}

具体策略对象存放在一个Map中,优化后的实现

Strategystrategy=map.get(param);
strategy.run();

上面这种优化方案有一个弊端,为了能够快速拿到对应的策略实现,需要map对象来保存策略,当添加一个新策略的时候,还需要手动添加到map中,容易被忽略。

2.2 枚举

发现很多同学不知道在枚举中可以定义方法,这里定义一个表示状态的枚举,另外可以实现一个run方法。

publicenumStatus{
NEW(0){
@Override
voidrun(){
//dosomething
}
},
RUNNABLE(1){
@Override
voidrun(){
//dosomething
}
};

publicintstatusCode;

abstractvoidrun();

Status(intstatusCode){
this.statusCode=statusCode;
}
}

重新定义策略枚举

publicenumStrategy{
FAST{
@Override
voidrun(){
//dosomething
}
},
NORMAL{
@Override
voidrun(){
//dosomething
}
},

SMOOTH{
@Override
voidrun(){
//dosomething
}
},

SLOW{
@Override
voidrun(){
//dosomething
}
};
abstractvoidrun();
}
通过枚举优化之后的代码如下
Strategystrategy=Strategy.valueOf(param);
strategy.run();

3. 学会使用 Optional

Optional主要用于非空判断,由于是jdk8新特性,所以使用的不是特别多,但是用起来真的爽。 使用之前:

if(user==null){
//doaction1
}else{
//doaction2
}
如果登录用户为空,执行action1,否则执行action 2,使用Optional优化之后,让非空校验更加优雅,间接的减少if操作
OptionaluserOptional=Optional.ofNullable(user);
userOptional.map(action1).orElse(action2);

4. 数组小技巧

来自google解释,这是一种编程模式,叫做表驱动法,本质是从表里查询信息来代替逻辑语句,比如有这么一个场景,通过月份来获取当月的天数,仅作为案例演示,数据并不严谨。 一般的实现:

intgetDays(intmonth){
if(month==1)return31;
if(month==2)return29;
if(month==3)return31;
if(month==4)return30;
if(month==5)return31;
if(month==6)return30;
if(month==7)return31;
if(month==8)return31;
if(month==9)return30;
if(month==10)return31;
if(month==11)return30;
if(month==12)return31;
}
优化后的代码
intmonthDays[12]={31,29,31,30,31,30,31,31,30,31,30,31};
intgetDays(intmonth){
returnmonthDays[--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_stat e这样的方法里面,然后通过单层的 switch 或者查找表来调用。

其实通常需要优化的都是整体接口抽象,而不是单个接口的实现,单个接口实现不清晰通常是因为接口实现和需求不同构造成的。






审核编辑:刘清

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • C语言
    +关注

    关注

    180

    文章

    7534

    浏览量

    128848
  • MAP
    MAP
    +关注

    关注

    0

    文章

    47

    浏览量

    15027
  • cache技术
    +关注

    关注

    0

    文章

    41

    浏览量

    986

原文标题:C语言中if、else如何进行优化分析

文章出处:【微信号:嵌入式开发爱好者,微信公众号:嵌入式开发爱好者】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    裁剪PCB多余的部分

    我设计了一块PCB但是一块多余的地方,那我要怎么裁剪?画黑线的地方,麻烦说的详细点 ,我尝试但没成功
    发表于 12-12 17:56

    看过来:嵌套的if…else…实现方法

    问题,而且就只用一个case框就能解决。一个button代表一个if...else...,用0和1表示先将这些button组成数组,然后转换成数值(在属性里可以找到数值的“二进制显示”方式)然后输入至case
    发表于 01-08 10:15

    CCSv4.2生成.hex文件,请问如何屏蔽多余byte?

    多余的byte。前面有8个,末尾2个。例如::20000000AA08000018001800000000000000000000000000F7AA0200000000004000E3AB8D实际的“:20000000"和"8D"是
    发表于 06-11 09:13

    如何优雅地完成倒计时定时器自适应显示

    如何实现倒计时的基本功能?如何优雅地完成倒计时定时器自适应显示
    发表于 10-27 07:15

    如何利用stm32芯片多余的IO来模拟串口

    如何利用stm32芯片多余的IO来模拟串口哪些优势?
    发表于 12-07 06:10

    CH32V103打开-flto选项后如何避免_write函数被优化

    _write函数被优化?或者是否还有其它方法可以用来减小代码体积?同样功能的源代码在ARM上面即使使用速度优化,FLASH空间也十分充足
    发表于 05-24 08:00

    如何优化RA2L1开发板上BSP不是很必要的部分

      0、起因  在上手RA2L1开发板的时候,使用RTT Studio给的BSP进行开发时发现默认占用太大了,所以今天就来说一下如何优化不是很必要的部分(当然芯片资源很大且不在乎的可以不做
    发表于 03-06 10:47

    always block內省略else所代表的電路 (SOC) (Verilog)

    Abstract在Verilog中,always block可以用來代表Flip-Flop, Combination Logic與Latch,本文比較在不寫else下,always block所代表
    的头像 发表于 09-28 11:16 4511次阅读
    always block內省略<b class='flag-5'>else</b>所代表的電路 (SOC) (Verilog)

    焊盘如何去除多余的锡?

    密集的微小无孔焊盘,比如芯片焊盘,可以采用吸锡铜带,配合松香,吸掉焊盘上多余的焊锡,也可以采用擦的方法,用电烙铁先将焊盘上的锡融化,再快速的用棉棍趁热擦去多余的锡。
    的头像 发表于 06-20 18:30 2.3w次阅读

    不会有人不知道怎么优雅的替换if-else语句吧

    又一层,虽然业务功能倒是实现了,但是看起来是真的很不优雅,尤其是对于我这种有强迫症的程序“猿”,看到这么多if-else,脑袋瓜子就嗡嗡的,总想着解锁新姿势:干掉过多的if-else!!!本文将介绍
    的头像 发表于 07-28 15:46 1211次阅读
    不会有人不知道怎么<b class='flag-5'>优雅</b>的替换if-<b class='flag-5'>else</b>语句吧

    利用Java 8的Function接口来消灭if...else

    在开发过程中经常会使用if...else...进行判断抛出异常、分支处理等操作。这些if...else...充斥在代码中严重影响了代码代码的美观,这时我们可以利用Java 8的Func
    的头像 发表于 04-21 10:23 2342次阅读

    关于Python中的“for-else”功能

    无论使用哪种编程语言,我们都会编写“if-else”语句,但是“for-else”呢?
    发表于 09-26 14:44 465次阅读

    代码如何优化多余的if/else?

    观点一(灵剑): 前期迭代懒得优化,来一个需求,加一个if,久而久之,就串成了一座金字塔。 当代码已经复杂到难以维护的程度之后,只能狠下心重构优化。那,有什么方案可以
    的头像 发表于 06-22 10:01 485次阅读
    代码如何<b class='flag-5'>优化</b>掉<b class='flag-5'>多余</b>的if/<b class='flag-5'>else</b>?

    如何解决冗长的if...else条件判断(上)

    : print ( "不知道是什么" ) # 写很长的一段if语句来判断不同的情况 这段代码的使用场景是满足用户可以完成在不同场景进行对应的操作,对应后端的代码,你肯定能想到最简单的实现方式就是上面的if ... else 语句。但是随着你的场景不断的增加
    的头像 发表于 09-12 17:03 407次阅读

    ttl门多余的输入端如何处理 ttl多余的输入端可以悬空吗

    ttl门多余的输入端如何处理 ttl多余的输入端可以悬空吗  TTL门是一种常见的数字逻辑门。TTL门通常具有多个输入端,其中有些输入端在特定的使用情况下可能是多余的。
    的头像 发表于 02-18 16:26 792次阅读