switch特性介绍

1、假设switch语句的分支比较少的时候(例如3,少于4的时候没有意义)没有必要使用此结构,相当于if。
2、各个分支常量的差值较大的时候,编译器会在效率还是内存进行取舍,这个时候编译器还是会编译成类似于if,else的结构。
3、在分支比较多的时候:在编译的时候会生成一个表(跳转表每个地址四个字节)。

switch汇编代码案例

三个及以下case

1.创建工程在main函数页面写下如下代码:

void funA(int a){switch (a) {case 1:printf("向上");break;case 2:printf("向下");break;case 3:printf("向左");break;default:printf("原地不动");break;}
}

2.Debug -> Debug Overflow -> Always show Disassembly 调成编译断点模式在方法处打断点,真机运行查看汇编代码。

switch 3个case汇编代码

3.上图基本流程如下(省略系统正常操作):

1.将参数a-1判断a和1是否相等
2.相等则执行case代码
3.不相等则将参数 a - 2再判断 a 和 2是否相等
4.根据case递增判断知道Default 结束

注:这和if else判断相似一个一个比较,是最基本的方法。

四个case

1.输入以下代码

void funA(int a){switch (a) {case 1:printf("向上");break;case 2:printf("向下");break;case 3:printf("向左");break;case 4:printf("向右");break;default:printf("原地不动");break;}
}

2.编译真机运行获得如下结果:

switch4选项以上

3.如上图解析:

1.先将参数减一和减四(4位case的数量)。
2.先判断是否为Default选项(上面减四就是为了判断是否为Default的情况)。
3.在物理地址部分建立内存表并且将case按顺序放进内存表中
4.获取内存地址,然后根据参数减一后偏移2个单位,查询case在表中的具体地址准确找到对应的地址内存并且将值赋给寄存器。
5.根据获得的地址跳转指定的case直接找到目标执行。

注:内存地址的偏移是从0开始,所以参数需要减一来适应index。

4.神奇的内存计算,下面附上内存计算过程的图片,因为重新执行程序左边的内存地址会有差异但是pc执行代码是一样的:

内存地址计算

汇编指令的用法见https://www.jianshu.com/p/583709763fa5

1.首先获取物理内存地址为0x1040ce8a8
2.根据参数传的是3,然后减一得2(10)偏移两个单位即1000为8位得到BC FF FF FF内存地址是从右往左读的
3.将获得的物理地址0x1040ce8a8加上计算得出的偏移地址后就等到了跳转执行地址(即在内存表格中确定具体内存位置)。
4.直接跳转执行。

注:计算方法为将物理地址0x1040ce8a8 + 0xffffffffbc = 0x1040ce8a8 - 0x44(取反加1 补码 详见补码) = 0x1040ce864 计算结果跟上图打印结果完全一致是不是很神奇?

分部差异大的例子

1.输入以下代码

void funA(int a){switch (a) {case 1:printf("向上");break;case 200:printf("向下");break;case 3000:printf("向左");break;case 178:printf("向右");break;default:printf("原地不动");break;}
}

2.编译真机运行获得如下结果:

跳跃性switch

3.解析:
当case的判断条件跳跃性太大,编译器就会变为if判断一样采用一一比较的方式进行判断效率是不高的。

所以在写switch语句的时候尽量要将其判断的case连续起来这样即减少运行时间又节省系统内存

执行switch时,会生成一张跳转表,表项数为(最大case值-最小case值+1),跳转表是一个数组,数组是一段连续的内存,jt数组中包含了7个表项(数组索引对应值),每个都是指向对应代码块的指针。
编译器将switch值n-最小case值(100),把取值范围移动到0至6之间,创建出一个新的程序变量index。首先判断index是否大于6,来判断是否在范围之外,如果超出直接执行default,即log_def指针对应代码块。
否则,根据索引的值直接跳转不同位置。

执行if-else是逐个条件进行判断,直到命中;与if-else语句相比,使用跳转表的优点是执行switch语句的时间与数量无关,且读取switch参数时只读取一次,就可跳到对应分支;缺点是维系了一个连续的数组,实际时使用空间换时间。

差异小的部分

这边简单介绍下,差异小的部分编译器会根据系统内存和时间之间取舍,比如4个case差值在8之间会建八个元素的表,刚才4个内存位,现在就是8个内存位,case以外的间隔默认为Default。具体的时间空间问题是编译器决定的。

总结

  • Switch在case连续的且大于等于4个的情况向下采用建表查询的方式,效率是大于if else语句的。
  • 小于3个case和case语句不规律差异较大建表需要耗费很大内存的情况下是相当于if else语句的。
  • 编译器是根据时间和空间的消耗来决定那种方式效率更高,所以在Switch写判断条件的时候最好做到连续紧密,可以最大限度的节省时间和内存。

原文链接

iOS Switch内部原理相关推荐

  1. iOS程序启动原理(上)

    为什么80%的码农都做不了架构师?>>>    iOS程序启动原理 Info.plist 常见设置 建立一个工程后,会在Supporting files文件夹下看到一个"工 ...

  2. JavaScriptCore内部原理(一):从JS源码到字节码的追踪

    一.概述 事实证明,在Fuzzing Webkit的过程中,使用Fuzzilli对JavaScriptCore(JSC)进行Fuzzing会非常成功,随着时间的推移,会产生大量崩溃.但是,一旦出现崩溃 ...

  3. iOS程序启动原理---iOS-Apple苹果官方文档翻译

    本系列所有开发文档翻译链接地址:iOS7开发-Apple苹果iPhone开发Xcode官方文档翻译PDF下载地址 //转载请注明出处--本文永久链接:http://www.cnblogs.com/Ch ...

  4. switch+ios风格android,安卓仿IOS Switch开关按钮

    老规矩,先放上一张图,对比一下安卓原生实现的Switch按钮和仿IOS实现的区别: Switch效果对比 下面主要讲讲仿IOS Switch控件的实现,其实安卓Switch控件中提供了两个属性让我们去 ...

  5. Git详解之九 Git内部原理

    以下内容转载自:http://www.open-open.com/lib/view/open1328070620202.html Git 内部原理 不管你是从前面的章节直接跳到了本章,还是读完了其余各 ...

  6. python array赋值_从踩坑学Python内部原理(5):执行时机的差异

    (给Python开发者加星标,提升Python技能) 英文:Satwik Kansal,翻译:暮晨 Python开发者整理自 GitHub [导读]:Python 是一个设计优美的解释型高级语言,它提 ...

  7. git gui fetch不到文件_Git内部原理剖析,有比这还详细的吗?

    1.1. 为什么写这篇文章 写这篇文章的本意有二: 工作安排原因,常有同事询问我一些关于 Git 的问题,总觉得自己解释的不够透彻,因此觉得有必要深入了解一下. 目前中文的 Git 教程往往本末倒置, ...

  8. JVM 内部原理(七)— Java 字节码基础之二

    JVM 内部原理(七)- Java 字节码基础之二 介绍 版本:Java SE 7 为什么需要了解 Java 字节码? 无论你是一名 Java 开发者.架构师.CxO 还是智能手机的普通用户,Java ...

  9. ORACLE 内部原理

    http://www.ohsdba.cn/index.php?m=Article&a=index&id=46 内部原理 2016-05-04• 如何使用BBED 2016-04-16• ...

最新文章

  1. 单例模式的3种实现方式, 及其性能对比
  2. 英国研发锡钙钛矿太阳能电池更高效环保
  3. 关于WSE_CLIPSIBLINGS
  4. 数据结构-joseph环
  5. JavaScript公共运行库
  6. php5.6 交叉编译,Cross-compile - 龙芯开源社区
  7. 分布式消息中间件-Rocketmq
  8. SAP Spartacus ComponentData的提前subscription
  9. PMP读书笔记(第2章)
  10. linux java weblogic,Linux java 安装问题
  11. UI设计灵感|挑战经典!不一样的注册登录页
  12. 年会聚会派对邀请海报素材模板,创意和心意缺一不可
  13. 零基础学python书籍-5本好书------Python零基础到入门必读Python书籍!
  14. maven中常遇到的一些错误
  15. Oracle JDBC 驱动
  16. 典型相关分析 CCA
  17. CSD-1371电力监控系统网络安全监测装置(Ⅱ 型)介绍
  18. voc旋转标注数据转dota类型
  19. Hbase——常见错误
  20. APS系统的实施步骤,外行人都能看懂

热门文章

  1. MNIST手写字体结构
  2. 从问卷设计流程看问卷质量的控制点
  3. monkey测试工具
  4. for循环嵌套的两种用法
  5. windows不是正版电脑壁纸变黑色
  6. 学习笔记—android实现opencv人脸检测
  7. 一个查看天气气流风向的网站,很炫
  8. Ansible 学习笔记(一)
  9. Photoshop入门 基本工具的使用方法
  10. java布尔变量_java布尔型变量理解