在学习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
树莓派文章专题:树莓派是什么?你不知道树莓派的常识和使用