您的位置 首页 制造

C言语的那些小秘密之变参函数的完成

在学习C语言的过程中我们可能很少会去写变参函数,印象中大学老师好像也没有提及过,但我发现变参函数的实现很巧妙,所以还是特地在此分析下变参函数的实现原理。无需标准C的支持,我们自己写代码来实现。

  在学习C言语的过程中咱们或许很少会去写变参函数,印象中大学老师如同也没有提及过,但我发现变参函数的完成很奇妙,所以仍是特别在此剖析下变参函数的完成原理。无需规范C的支撑,咱们自己写代码来完成。

  先来看看一个完成代码:

  #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("%d\n", sum(4, 100,100,100,100));

  printf("%d\n", sum(3, 200, 200, 200));

  return 0;

  }

  运转成果如下:

  #define va_list void*经过这句代码咱们完成了界说va_list是一个指针,参数类型不定,它能够指向恣意类型的指针。为了让arg指向第一个可变参数,咱们用nr的地址加上nr的数据类型巨细就行了,选用如下的界说能够完成。

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

  经过(((char*)&(start)) + sizeof(start)) 能够得到第一个可变参数的地址,再将其强制转换为va_list类型。

  成功取出了第一个可变参数后,接下来的使命便是持续取出可变参数,办法跟上面求第一个可变参数的办法相同,经过arg = (char*)arg + sizeof(type);来完成让arg指向下一个可变参数,type为可变参数的类型,经过这种办法能够逐个取出可变参数。

  在这里趁便给出上面完成代码的汇编代码,有爱好的能够读读,加深下关于底层汇编代码的阅览才能。

  .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 "%d\n"

  .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

树莓派文章专题:树莓派是什么?你不知道树莓派的常识和使用

声明:本文内容来自网络转载或用户投稿,文章版权归原作者和原出处所有。文中观点,不代表本站立场。若有侵权请联系本站删除(kf@86ic.com)https://www.86ic.net/bandaoti/zhizao/116801.html

为您推荐

联系我们

联系我们

在线咨询: QQ交谈

邮箱: kf@86ic.com

关注微信
微信扫一扫关注我们

微信扫一扫关注我们

返回顶部