`
maomaozgc
  • 浏览: 329115 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

switch和if-else相比

阅读更多
switch和if-else相比,由于使用了Binary Tree算法,绝大部分情况下switch会快一点,除非是if-else的第一个条件就为true.
说实话  我也没有深入研究过这个问题的根源
只是在实际开发中  没有人会去用很多很多else if的
都是用 switch case 的  后者比较清晰  给人感觉就是一个脑子很清楚的人写出来的东西
至于效率的本质  就让大企鹅去操心吧

编译器编译switch与编译if...else...不同。不管有多少case,都直接跳转,不需逐个比较查询。

昨天发现了一本叫做CSAPP的书,终于找到了关于switch问题的解答。
这是一段C代码:
/* $begin switch-c */
int switch_eg(int x)
{
    int result = x;

    switch (x) {

    case 100:
    result *= 13;
    break;

    case 102:
    result += 10;
    /* Fall through */

    case 103:
    result += 11;
    break;

    case 104:
    case 106:
    result *= result;
    break;

    default:
    result = 0;      
    }

    return result;
}
/* $end switch-c */

用GCC汇编出来的代码如下:
    .file    "switch.c"
    .version    "01.01"
gcc2_compiled.:
.text
    .align 4
.globl switch_eg
    .type     switch_eg,@function
switch_eg:
    pushl %ebp
    movl %esp,%ebp
    movl 8(%ebp),%edx
    leal -100(%edx),%eax
    cmpl ,%eax
    ja .L9
    jmp *.L10(,%eax,4)
    .p2align 4,,7
.section    .rodata
    .align 4
    .align 4
.L10:
    .long .L4
    .long .L9
    .long .L5
    .long .L6
    .long .L8
    .long .L9
    .long .L8
.text
    .p2align 4,,7
.L4:
    leal (%edx,%edx,2),%eax
    leal (%edx,%eax,4),%edx
    jmp .L3
    .p2align 4,,7
.L5:
    addl ,%edx
.L6:
    addl ,%edx
    jmp .L3
    .p2align 4,,7
.L8:
    imull %edx,%edx
    jmp .L3
    .p2align 4,,7
.L9:
    xorl %edx,%edx
.L3:
    movl %edx,%eax
    movl %ebp,%esp
    popl %ebp
    ret
.Lfe1:
    .size     switch_eg,.Lfe1-switch_eg
    .ident    "GCC: (GNU) 2.95.3 20010315 (release)"

在上面的汇编代码中我们可以很清楚的看到switch部分被分配了一个连续的查找表,switch case中不连续的部分也被添加上了相应的条目,switch表的大小不是根据case语句的多少,而是case的最大值的最小值之间的间距。在选择相应 的分支时,会先有一个cmp子句,如果大于查找表的最大值,则跳转到default子句。而其他所有的case语句的耗时都回事O(1)。

相比于if-else结构,switch的效率绝对是要高很多的,但是switch使用查找表的方式决定了case的条件必须是一个连续的常量。而if-else则可以灵活的多。


可以看到if-else只是单纯地一个接一个比较,效率比较低

可以看出,switch的效率一般比if-else高


switch   效率高,     从汇编代码可以看出来  
   
  switch   只计算一次值   然后都是test   ,   jmp,    
   
  if...else   是每个条件都要计算一遍的. 


  switch的效率与分支数无关  
  当只有分支比较少的时候,if效率比switch高(因为switch有跳转表)  
  分支比较多,那当然是switch
分享到:
评论
2 楼 xfjt297857539 2011-08-19  
不错的!不错的!不错的!
1 楼 1927105 2011-04-28  
学习了,之前看过,在复习一下

相关推荐

    go-hsm:实现分层状态机(HSM)的 Golang 库

    与传统的状态机实现方法(例如嵌套 if-else/switch、状态表、OOP 中的状态设计模式)相比,HSM 提供了以下主要优势: 它支持嵌套状态和 hehavior 继承 它为状态提供进入和退出动作 它使用类层次结构来表示状态层次...

    casematch:Java 8的Matcher库

    与if-then-else级联相比,它允许更具可读性的代码,但是会增加运行时和内存开销。 该库将不提供像从功能语言(或具有更多功能方面的语言)(例如Scala)中已知的模式匹配一​​样强大和紧凑的匹配。 尤其是缺少对象...

    C语言程序设计标准教程

     if语句,switch语句 (2) 循环执行语句  do while语句,while语句,for语句 (3) 转向语句  break语句,goto语句,continue语句,return语句 4.复合语句 把多个语句用括号{}括起来组成的一个语句称复合语句。 在...

    怎么刷leetcode-100-days-of-code:扩展JavaScript知识

    If/Else 语句、Switch 语句甚至三元运算符的所有信息。 还学习了关于Truthy 和fasly 以及平等的知识。 感想:首先想说,今天开始做100天的编码挑战 :beaming_face_with_smiling_eyes: . 大喊 Hugo P. 让我开始。 ...

    uboott移植实验手册及技术文档

    switch (s) { case NFCE_LOW: nand->NFCONF &= ~(1); break; case NFCE_HIGH: nand->NFCONF |= (1); Create PDF files without this message by purchasing novaPDF printer (http://www.novapdf.com) ...

    Visual C++ 2005入门经典--源代码及课后练习答案

    3.1.4 嵌套的if-else语句 107 3.1.5 逻辑运算符和表达式 109 3.1.6 条件运算符 112 3.1.7 switch语句 114 3.1.8 无条件转移 116 3.2 重复执行语句块 117 3.2.1 循环的概念 117 3.2.2 for循环的变体 ...

    Visual C++ 2010入门经典(第5版)--源代码及课后练习答案

    3.1.3 嵌套的if-else语句 107 3.1.4 逻辑运算符和表达式 109 3.1.5 条件运算符 112 3.1.6 switch语句 113 3.1.7 无条件转移 116 3.2 重复执行语句块 117 3.2.1 循环的概念 117 3.2.2 for循环的变体 119 ...

    as3 接口类的用法和好处

    原来,接口和抽象类相比,多出的一个优势在于(仅限Java和AS3),一个类可以实现多个接口,但是不能继承多个类。所以,如果在这里改用接口,就一切都好解决了。先定义两个接口: package{ public interface ...

    net学习笔记及其他代码应用

    1. 简述 private、 protected...因此传递给 switch 和 case 语句的参数应该是 int、 short、 char 或者 byte。long,string 都不能作用于swtich。 47.当一个线程进入一个对象的一个synchronized方法后,其它线程是否可...

    (谭浩强)c语言学习书

    signed static sizof struct switch typedef union unsigned void volatile while 9种控制语句,程序书写自由,主要用小写字母表示,压缩了一切不必要的成分。 Turbo C扩充了11个关键字: asm _cs _ds...

    C#坦克大战代码(电脑版本跟手机版本)

    程序很简单,跟java代码相比没有多大改动 开发环境 vs2008 实现方法如下 1.在form中添加一个panel,在panel的 Paint方法中得到Graphics对象 2.通过Graphics对象再panel画出坦克,子弹等相关内容 3.添加timer控件 ...

    简单C#坦克大战源码(电脑版本跟手机版本)

    程序很简单,跟java代码相比没有多大改动 实现方法如下 1.在form中添加一个panel,在panel的 Paint方法中得到Graphics对象 2.通过Graphics对象再panel画出坦克,子弹等相关内容 3.添加timer控件 来控制panel的重画 ...

    网管教程 从入门到精通软件篇.txt

    AIF,AIFF:音频互交换文件,Silicon Graphic and Macintosh应用程序的声音格式 ANI:Windows系统中的动画光标 ARC:LH ARC的压缩档案文件 ARJ:Robert Jung ARJ压缩包文件 ASD:Microsoft Word的自动保存文件;...

    c#学习笔记.txt

    选择语句if, else, switch, case 迭代语句do, for, foreach, in, while 跳转语句break, continue, default, goto, return 异常处理语句throw, try-catch, try-finally Checked 和 Uncheckedchecked, unchecked fixed...

    Java精华(免费版)

     相比之下,当你调用method2()时,则不需要把它放在try/catch程序块当中,因为它会抛出的exception不是checked exception,而是runtime exception。会抛出runtime exception的方法在定义时不必声明它会抛出...

Global site tag (gtag.js) - Google Analytics