成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

【揭秘】為什么switch...case比if...else執行效率高

開發 后端
在C語言中,條件判斷語句是程序的重要組成部分,也是系統業務邏輯的控制手段,教科書告訴我們switch...case...語句比if...else if...else執行效率要高。本文嘗試從匯編的角度予以分析并揭曉其中的奧秘。

 [[333956]]

switch...case與if...else的根本區別

switch...case會生成一個跳轉表來指示實際的case分支的地址,而這個跳轉表的索引號與switch變量的值是相等的。從而,switch...case不用像if...else那樣遍歷條件分支直到命中條件,而只需訪問對應索引號的表項從而到達定位分支的目的。

具體地說,switch...case會生成一份大小(表項數)為最大case常量+1的跳表,程序首先判斷switch變量是否大于最大case 常量,若大于,則跳到default分支處理;否則取得索引號為switch變量大小的跳表項的地址(即跳表的起始地址+表項大小*索引號),程序接著跳到此地址執行,到此完成了分支的跳轉。

第一步,寫一個demo程序:foo.c 

  1. #include <stdio.h>  
  2. static int  
  3. foo_ifelse(char c)  
  4.  
  5.         if (c == '0' || c == '1') {  
  6.                 c += 1;  
  7.         } else if (c == 'a' || c == 'b') {  
  8.                 c += 2;  
  9.         } else if (c == 'A' || c == 'B') {  
  10.                 c += 3;  
  11.         } else {  
  12.                 c += 4;  
  13.         }  
  14.         return (c);  
  15.  
  16. static int  
  17. foo_switch(char c)  
  18.  
  19.         switch (c) {  
  20.                 case '1':  
  21.                 case '0': c += 1; break;  
  22.                 case 'b':  
  23.                 case 'a': c += 2; break;  
  24.                 case 'B':  
  25.                 case 'A': c += 3; break;  
  26.                 default:  c += 4; break;  
  27.         }    
  28.         return (c); 
  29.  
  30. int  
  31. main(int argc, char **argv)  
  32.  
  33.         int m1 = foo_ifelse('0');  
  34.         int m2 = foo_ifelse('1');  
  35.         int n1 = foo_switch('a'); 
  36.         int n2 = foo_switch('b');  
  37.         (void) printf("%c %c %c %c\n", m1, m2, n1, n2);  
  38.         return (0);  

第二步,在Ubuntu上使用gcc編譯

$ gcc -g -o foo foo.c

第三步,使用gdb對二進制文件foo反匯編 (使用intel語法) 

  1. o 反匯編foo_ifelse() 
  2. (gdb) set disassembly-flavor intel  
  3. (gdb) disas /m foo_ifelse  
  4. Dump of assembler code for function foo_ifelse:  
  5. 4       {  
  6.    0x0804841d <+0>:     push   ebp  
  7.    0x0804841e <+1>:     mov    ebp,esp  
  8.    0x08048420 <+3>:     sub    esp,0x4  
  9.    0x08048423 <+6>:     mov    eax,DWORD PTR [ebp+0x8]  
  10.    0x08048426 <+9>:     mov    BYTE PTR [ebp-0x4],al    
  11. 5               if (c == '0' || c == '1') {  
  12.    0x08048429 <+12>:    cmp    BYTE PTR [ebp-0x4],0x30  
  13.    0x0804842d <+16>:    je     0x8048435 <foo_ifelse+24>  
  14.    0x0804842f <+18>:    cmp    BYTE PTR [ebp-0x4],0x31  
  15.    0x08048433 <+22>:    jne    0x8048441 <foo_ifelse+36>  
  16. 6                       c += 1;  
  17.    0x08048435 <+24>:    movzx  eax,BYTE PTR [ebp-0x4]  
  18.    0x08048439 <+28>:    add    eax,0x1  
  19.    0x0804843c <+31>:    mov    BYTE PTR [ebp-0x4],al  
  20.    0x0804843f <+34>:    jmp    0x804847b <foo_ifelse+94>  
  21. 7               } else if (c == 'a' || c == 'b') {  
  22.    0x08048441 <+36>:    cmp    BYTE PTR [ebp-0x4],0x61  
  23.    0x08048445 <+40>:    je     0x804844d <foo_ifelse+48>  
  24.    0x08048447 <+42>:    cmp    BYTE PTR [ebp-0x4],0x62  
  25.    0x0804844b <+46>:    jne    0x8048459 <foo_ifelse+60>  
  26. 8                       c += 2;  
  27.    0x0804844d <+48>:    movzx  eax,BYTE PTR [ebp-0x4]  
  28.    0x08048451 <+52>:    add    eax,0x2  
  29.    0x08048454 <+55>:    mov    BYTE PTR [ebp-0x4],al  
  30.    0x08048457 <+58>:    jmp    0x804847b <foo_ifelse+94>  
  31. 9               } else if (c == 'A' || c == 'B') {  
  32.    0x08048459 <+60>:    cmp    BYTE PTR [ebp-0x4],0x41  
  33.    0x0804845d <+64>:    je     0x8048465 <foo_ifelse+72>  
  34.    0x0804845f <+66>:    cmp    BYTE PTR [ebp-0x4],0x42  
  35.    0x08048463 <+70>:    jne    0x8048471 <foo_ifelse+84>  
  36. 10                      c += 3;  
  37.    0x08048465 <+72>:    movzx  eax,BYTE PTR [ebp-0x4]  
  38.    0x08048469 <+76>:    add    eax,0x3  
  39.    0x0804846c <+79>:    mov    BYTE PTR [ebp-0x4],al  
  40.    0x0804846f <+82>:    jmp    0x804847b <foo_ifelse+94>  
  41. 11              } else {  
  42. 12                      c += 4;  
  43.    0x08048471 <+84>:    movzx  eax,BYTE PTR [ebp-0x4]  
  44.    0x08048475 <+88>:    add    eax,0x4  
  45.    0x08048478 <+91>:    mov    BYTE PTR [ebp-0x4],al  
  46. 13              }  
  47. 14  
  48. 15              return (c);  
  49.    0x0804847b <+94>:    movsx  eax,BYTE PTR [ebp-0x4]  
  50. 16      }  
  51.    0x0804847f <+98>:    leave  
  52.    0x08048480 <+99>:    ret  
  53. End of assembler dump.  
  54. (gdb)o 反匯編foo_ifelse()  
  55. (gdb) set disassembly-flavor intel  
  56. (gdb) disas /m foo_ifelse  
  57. Dump of assembler code for function foo_ifelse:  
  58. 4       {  
  59.    0x0804841d <+0>:     push   ebp  
  60.    0x0804841e <+1>:     mov    ebp,esp  
  61.    0x08048420 <+3>:     sub    esp,0x4  
  62.    0x08048423 <+6>:     mov    eax,DWORD PTR [ebp+0x8]  
  63.    0x08048426 <+9>:     mov    BYTE PTR [ebp-0x4],al  
  64. 5               if (c == '0' || c == '1') {  
  65.    0x08048429 <+12>:    cmp    BYTE PTR [ebp-0x4],0x30  
  66.    0x0804842d <+16>:    je     0x8048435 <foo_ifelse+24>  
  67.    0x0804842f <+18>:    cmp    BYTE PTR [ebp-0x4],0x31  
  68.    0x08048433 <+22>:    jne    0x8048441 <foo_ifelse+36>  
  69. 6                       c += 1;  
  70.    0x08048435 <+24>:    movzx  eax,BYTE PTR [ebp-0x4]  
  71.    0x08048439 <+28>:    add    eax,0x1  
  72.    0x0804843c <+31>:    mov    BYTE PTR [ebp-0x4],al  
  73.    0x0804843f <+34>:    jmp    0x804847b <foo_ifelse+94>  
  74. 7               } else if (c == 'a' || c == 'b') {  
  75.    0x08048441 <+36>:    cmp    BYTE PTR [ebp-0x4],0x61  
  76.    0x08048445 <+40>:    je     0x804844d <foo_ifelse+48>  
  77.    0x08048447 <+42>:    cmp    BYTE PTR [ebp-0x4],0x62  
  78.    0x0804844b <+46>:    jne    0x8048459 <foo_ifelse+60>  
  79. 8                       c += 2;  
  80.    0x0804844d <+48>:    movzx  eax,BYTE PTR [ebp-0x4]  
  81.    0x08048451 <+52>:    add    eax,0x2  
  82.    0x08048454 <+55>:    mov    BYTE PTR [ebp-0x4],al  
  83.    0x08048457 <+58>:    jmp    0x804847b <foo_ifelse+94>  
  84. 9               } else if (c == 'A' || c == 'B') {  
  85.    0x08048459 <+60>:    cmp    BYTE PTR [ebp-0x4],0x41  
  86.    0x0804845d <+64>:    je     0x8048465 <foo_ifelse+72>  
  87.    0x0804845f <+66>:    cmp    BYTE PTR [ebp-0x4],0x42  
  88.    0x08048463 <+70>:    jne    0x8048471 <foo_ifelse+84> 
  89. 10                      c += 3;  
  90.    0x08048465 <+72>:    movzx  eax,BYTE PTR [ebp-0x4]  
  91.    0x08048469 <+76>:    add    eax,0x3  
  92.    0x0804846c <+79>:    mov    BYTE PTR [ebp-0x4],al  
  93.    0x0804846f <+82>:    jmp    0x804847b <foo_ifelse+94>  
  94. 11              } else {  
  95. 12                      c += 4;  
  96.    0x08048471 <+84>:    movzx  eax,BYTE PTR [ebp-0x4]  
  97.    0x08048475 <+88>:    add    eax,0x4  
  98.    0x08048478 <+91>:    mov    BYTE PTR [ebp-0x4],al  
  99. 13              }  
  100. 14  
  101. 15              return (c);  
  102.    0x0804847b <+94>:    movsx  eax,BYTE PTR [ebp-0x4]  
  103. 16      }  
  104.    0x0804847f <+98>:    leave  
  105.    0x08048480 <+99>:    ret  
  106. End of assembler dump.  
  107. (gdb) 

o 反匯編foo_switch() 

  1. (gdb) set disassembly-flavor intel  
  2. (gdb) disas /m foo_switch  
  3. Dump of assembler code for function foo_switch:  
  4. 20      {  
  5.    0x08048481 <+0>:     push   ebp  
  6.    0x08048482 <+1>:     mov    ebp,esp  
  7.    0x08048484 <+3>:     sub    esp,0x4  
  8.    0x08048487 <+6>:     mov    eax,DWORD PTR [ebp+0x8]  
  9.    0x0804848a <+9>:     mov    BYTE PTR [ebp-0x4],al  
  10. 21              switch (c) {  
  11.    0x0804848d <+12>:    movsx  eax,BYTE PTR [ebp-0x4]  
  12.    0x08048491 <+16>:    sub    eax,0x30  
  13.    0x08048494 <+19>:    cmp    eax,0x32  
  14.    0x08048497 <+22>:    ja     0x80484c6 <foo_switch+69>  
  15.    0x08048499 <+24>:    mov    eax,DWORD PTR [eax*4+0x80485f0]  
  16.    0x080484a0 <+31>:    jmp    eax  
  17. 22                      case '1':  
  18. 23                      case '0': c += 1; break;  
  19.    0x080484a2 <+33>:    movzx  eax,BYTE PTR [ebp-0x4]  
  20.    0x080484a6 <+37>:    add    eax,0x1  
  21.    0x080484a9 <+40>:    mov    BYTE PTR [ebp-0x4],al  
  22.    0x080484ac <+43>:    jmp    0x80484d1 <foo_switch+80>  
  23. 24                      case 'b':  
  24. 25                      case 'a': c += 2; break;  
  25.    0x080484ae <+45>:    movzx  eax,BYTE PTR [ebp-0x4]  
  26.    0x080484b2 <+49>:    add    eax,0x2  
  27.    0x080484b5 <+52>:    mov    BYTE PTR [ebp-0x4],al  
  28.    0x080484b8 <+55>:    jmp    0x80484d1 <foo_switch+80>  
  29. 26                      case 'B':  
  30. 27                      case 'A': c += 3; break;  
  31.    0x080484ba <+57>:    movzx  eax,BYTE PTR [ebp-0x4]  
  32.    0x080484be <+61>:    add    eax,0x3  
  33.    0x080484c1 <+64>:    mov    BYTE PTR [ebp-0x4],al  
  34.    0x080484c4 <+67>:    jmp    0x80484d1 <foo_switch+80>  
  35. 28                      default:  c += 4; break;  
  36.    0x080484c6 <+69>:    movzx  eax,BYTE PTR [ebp-0x4]  
  37.    0x080484ca <+73>:    add    eax,0x4  
  38.    0x080484cd <+76>:    mov    BYTE PTR [ebp-0x4],al  
  39.    0x080484d0 <+79>:    nop  
  40. 29              } 
  41. 30  
  42. 31              return (c);  
  43.    0x080484d1 <+80>:    movsx  eax,BYTE PTR [ebp-0x4]  
  44. 32      }  
  45.    0x080484d5 <+84>:    leave  
  46.    0x080484d6 <+85>:    ret  
  47. End of assembler dump.  
  48. (gdb) 

分析:

  •  在foo_ifelse()中,采用的方法是按順序比較,如滿足條件,則執行對應的代碼,否則跳轉到下一個分支再進行比較;
  •  在foo_switch()中,下面的這段匯編代碼比較有意思, 
  1. ..  
  2. 21 switch (c) {  
  3.    0x0804848d <+12>:    movsx  eax,BYTE PTR [ebp-0x4]  
  4.    0x08048491 <+16>:    sub    eax,0x30  
  5.    0x08048494 <+19>:    cmp    eax,0x32  
  6.    0x08048497 <+22>:    ja     0x80484c6 <foo_switch+69>  
  7.    0x08048499 <+24>:    mov    eax,DWORD PTR [eax*4+0x80485f0]  
  8.    0x080484a0 <+31>:    jmp    eax  
  9. .. 

注意: 

第17行 jmp eax

也就是說,當c的取值不同,是什么機制保證第17行能跳轉到正確的位置開始執行呢?

第16行: eax = [eax * 4 + 0x80485f0]

搞清楚了從地址0x80485f0開始,對應的內存里面的內容也就回答了剛才的問題。

執行完第16行后,

  •  當c為'1'或'0'時, eax的值應該是0x080484a2;
  •  當c為'b'或'a'時, eax的值應該是0x080484ae;
  •  當c為'B'或'A'時, eax的值應該是0x080484ba;

通過gdb查看對應的內存,確實如此! 

  1. >>> ord('1') - 0x30  
  2. >>> ord('0') - 0x30  
  3. (gdb) x /2wx  0*4+0x80485f0  
  4. 0x80485f0:    0x080484a2    0x080484a2  
  5. >>> ord('b') - 0x30  
  6. >>> ord('a') - 0x30  
  7. (gdb) x /2wx 49*4+0x80485f0  
  8. 0x80486b4:    0x080484ae    0x080484ae             
  9. >>> ord('B') - 0x30  
  10. >>> ord('A') - 0x30  
  11. (gdb) x /2wx 17*4+0x80485f0  
  12. 0x8048634:    0x080484ba    0x080484ba 

那么,我們可以大膽的猜測,雖然c的取值不同但是跳轉的IP確實是精準無誤的,一定是編譯階段就被設定好了,果真如此嗎?接下來分析一下對應的二進制文件foo,

第四步,使用objdump查看foo, 

  1. $ objdump -D foo > /tmp/x  
  2. $ vim /tmp/x  
  3.  509 Disassembly of section .rodata:  
  4.  ...  
  5.  518  80485f0:       a2 84 04 08 a2          mov    %al,0xa2080484  
  6.  519  80485f5:       84 04 08                test   %al,(%eax,%ecx,1)  
  7.  ...  
  8.  534  8048630:       c6 84 04 08 ba 84 04    movb   $0x8,0x484ba08(%esp,%eax,1)  
  9.  535  8048637:       08  
  10.  536  8048638:       ba 84 04 08 c6          mov    $0xc6080484,%edx  
  11.  ...  
  12.  566  80486b0:       c6 84 04 08 ae 84 04    movb   $0x8,0x484ae08(%esp,%eax,1)  
  13.  567  80486b7:       08  
  14.  568  80486b8:       ae                      scas   %es:(%edi),%al  
  15.  569  80486b9:       84 04 08                test   %al,(%eax,%ecx,1)  
  16.  ... 

在0x80485f0地址,存的8個字節正好是0x080484a2, 0x080484a2 (注意:按照小端的方式閱讀)

在0x80486b4地址,存的8個字節正好是0x080484ae, 0x080484ae

在0x8048634地址,存的8個字節正好是0x080484ba,0x080484ba

果然不出所料,要跳轉的IP的值正是在編譯的時候存入了.rodata(只讀數據區)。一旦foo開始運行,對應的內存地址就填寫上了正確的待跳轉地址,接下來只不過是根據c的取值計算出對應的IP存放的內存起始地址X,從X中取出待跳轉的地址,直接跳轉就好。 

  1. 16    0x08048499 <+24>:    mov    eax,DWORD PTR [eax*4+0x80485f0]  
  2. 17    0x080484a0 <+31>:    jmp    eax 

到此為止,我們已經搞清楚了為什么switch...case...語句相對于if...else if...else...來說執行效率要高的根本原因。簡言之,編譯的時候創建了一個map存于.rodata區中,運行的時候直接根據輸入(c的值)查表,找到對應的IP后直接跳轉。(省去了cmp, jmp -> cmp, jmp -> cmp, jmp...這一冗長的計算過程。)

總結:

switch...case...執行效率高,屬于典型的以空間換時間。也就是說,(套用算法的行話)以提高空間復雜度為代價降低了時間復雜度。

 

【責任編輯:龐桂玉 TEL:(010)68476606】

 

責任編輯:龐桂玉 來源: C語言與C++編程
相關推薦

2011-05-25 14:59:35

if elseswitch case

2023-06-07 08:35:36

2019-09-11 09:09:56

++ii++編程語言

2021-07-21 09:35:36

switchbreakJava

2019-07-05 16:26:06

MySQLcount(1)count(*)

2011-09-13 09:57:25

谷歌云計算

2025-04-21 00:00:05

2011-04-06 14:20:50

Java編程

2022-10-17 08:03:54

CPUDMAKafka

2011-04-13 09:13:02

Java內存

2009-08-19 10:41:14

C# switch和c

2012-03-12 11:48:44

惠普激光打印機

2023-07-26 07:02:04

2013-01-18 11:16:15

效率

2021-01-13 10:51:08

PromissetTimeout(函數

2022-11-10 15:32:29

2020-01-15 14:20:07

Node.js應用程序javascript

2022-05-31 14:43:47

微軟AI研究

2022-09-16 15:02:19

戴爾

2011-04-25 17:04:28

傳真機
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 国产福利在线 | 怡红院怡春院一级毛片 | 日本黄色不卡视频 | av大片在线观看 | 久久51 | 欧美aaa| 人和拘一级毛片c | 妖精视频一区二区三区 | 日本精品一区二区三区在线观看 | 免费黄色a级毛片 | 欧美视频网 | 中文字幕一区二区三区精彩视频 | 日韩在线免费视频 | 欧美国产日本一区 | 黄色成人亚洲 | 国产精品无码久久久久 | 亚洲国产精品网站 | 日日草夜夜草 | 日本综合在线观看 | 中文字幕在线一区 | 成人久久久久久久久 | 黄色一级片在线播放 | 在线一区| 国产高清区 | 亚洲不卡在线观看 | 日本福利在线 | 国产精品久久久久久久久久久新郎 | 国产精品99久久久久久大便 | 国产欧美精品 | 一级黄色网页 | 国产一区二区三区在线看 | a在线观看免费 | 久国久产久精永久网页 | 午夜免费视频 | 欧美男人天堂 | 精品国产免费人成在线观看 | 一级片av| 狠狠色狠狠色综合系列 | 日韩在线一区二区 | 欧美精品在线免费观看 | 亚洲欧美日韩久久久 |