新聞中心

EEPW首頁 > 嵌入式系統(tǒng) > 牛人業(yè)話 > C語言的那些小秘密之變參函數(shù)的實(shí)現(xiàn)

C語言的那些小秘密之變參函數(shù)的實(shí)現(xiàn)

作者: 時(shí)間:2015-03-06 來源:網(wǎng)絡(luò) 收藏

  在學(xué)習(xí)C語言的過程中我們可能很少會(huì)去寫變參,印象中大學(xué)老師好像也沒有提及過,但我發(fā)現(xiàn)變參的實(shí)現(xiàn)很巧妙,所以還是特地在此分析下變參的實(shí)現(xiàn)原理。無需標(biāo)準(zhǔn)C的支持,我們自己寫代碼來實(shí)現(xiàn)。

本文引用地址:http://2s4d.com/article/270588.htm

  先來看看一個(gè)實(shí)現(xiàn)代碼:

  #include

  #define va_list void*

  #define va_arg(arg, type) *(type*)arg; arg = (char*)arg + sizeof(type);

  #define va_start(arg, start) arg = (va_list)(((char*)&(start)) + sizeof(start))

  int sum(int nr, ...)

  {

  int i = 0;

  int result = 0;

  va_list arg = NULL;

  va_start(arg, nr);

  for(i = 0; i < nr; i++)

  {

  result += va_arg(arg, int);

  }

  return result;

  }

  int main(int argc, char* argv[])

  {

  printf("%dn", sum(4, 100,100,100,100));

  printf("%dn", sum(3, 200, 200, 200));

  return 0;

  }

  運(yùn)行結(jié)果如下:



  #define va_list void*通過這句代碼我們實(shí)現(xiàn)了定義va_list是一個(gè)指針,參數(shù)類型不定,它可以指向任意類型的指針。為了讓arg指向第一個(gè)可變參數(shù),我們用nr的地址加上nr的數(shù)據(jù)類型大小就行了,采用如下的定義可以實(shí)現(xiàn)。

  #define va_start(arg, start) arg = (va_list)(((char*)&(start)) + sizeof(start)) 。

  通過(((char*)&(start)) + sizeof(start)) 可以得到第一個(gè)可變參數(shù)的地址,再將其強(qiáng)制轉(zhuǎn)換為va_list類型。

  成功取出了第一個(gè)可變參數(shù)后,接下來的任務(wù)就是繼續(xù)取出可變參數(shù),方法跟上面求第一個(gè)可變參數(shù)的方法一樣,通過arg = (char*)arg + sizeof(type);來實(shí)現(xiàn)讓arg指向下一個(gè)可變參數(shù),type為可變參數(shù)的類型,通過這種方法可以一一取出可變參數(shù)。

  在這里順便給出上面實(shí)現(xiàn)代碼的匯編代碼,有興趣的可以讀讀,加深下對(duì)于底層匯編代碼的閱讀能力。

  .file "varargs.c"

  .text

  .globl sum

  .type sum, @function

  sum:

  pushl %ebp

  movl %esp, %ebp

  subl $16, %esp

  movl $0, -4(%ebp)

  movl $0, -8(%ebp)

  movl $0, -12(%ebp)

  leal 12(%ebp), %eax

  movl %eax, -12(%ebp)

  movl $0, -4(%ebp)

  jmp .L2

  .L3:

  movl -12(%ebp), %eax

  movl (%eax), %eax

  addl %eax, -8(%ebp)

  addl $4, -12(%ebp)

  addl $1, -4(%ebp)

  .L2:

  movl 8(%ebp), %eax

  cmpl %eax, -4(%ebp)

  jl .L3

  movl -8(%ebp), %eax

  leave

  ret

  .size sum, .-sum

  .section .rodata

  .LC0:

  .string "%dn"

  .text

  .globl main

  .type main, @function

  main:

  pushl %ebp

  movl %esp, %ebp

  andl $-16, %esp

  subl $32, %esp

  movl $100, 16(%esp)

  movl $100, 12(%esp)

  movl $100, 8(%esp)

  movl $100, 4(%esp)

  movl $4, (%esp)

  call sum

  movl $.LC0, %edx

  movl %eax, 4(%esp)

  movl %edx, (%esp)

  call printf

  movl $200, 12(%esp)

  movl $200, 8(%esp)

  movl $200, 4(%esp)

  movl $3, (%esp)

  call sum

  movl $.LC0, %edx

  movl %eax, 4(%esp)

  movl %edx, (%esp)

  call printf

  movl $0, %eax

  leave

  ret

  .size main, .-main

  .ident "GCC: (Ubuntu/Linaro 4.5.2-8ubuntu4) 4.5.2"

  .section .note.GNU-stack,"",@progbits

樹莓派文章專題:樹莓派是什么?你不知道樹莓派的知識(shí)和應(yīng)用

c語言相關(guān)文章:c語言教程



上一頁 1 2 下一頁

關(guān)鍵詞: C語言 函數(shù)

評(píng)論


相關(guān)推薦

技術(shù)專區(qū)

關(guān)閉