va_list、 va_start、 vsprintf、va_end 用法

来源:互联网 发布:淘宝详情图片模板 编辑:程序博客网 时间:2024/05/16 07:52

小知识点总结:(知识点2、3是为了更好理解知识点1)

知识点1:va_list、 va_start、 vsprintf、va_end 用法: 

看同事LCD工程里面有这样的应用--

va_list arg_ptr;
    
va_start(arg_ptr, format);

slen = vsprintf(buf, format, arg_ptr);

va_end(arg_ptr);


百度后的最容易理解的解析:--用法

       1)首先在函数里定义一具VA_LIST型的变量,这个变量是指向参数的指针

      2)然后用VA_START宏初始化变量刚定义的VA_LIS T变量,这个宏的第二个参数是第一个可变参数的前一个参数,是一个固定的参数。

       3)然后用VA_ARG返回可变的参数,VA_ARG的第二个参数是你要返回的参数的类型。

       4)最后用VA_END宏结束可变参数的获取。然后你就可以在函数里使用第二个参数了。如果函数有多个可变参数的,依次调用VA_ARG获取各个参数。

实例:

/* 函数名: vsprintf功 能: 送格式化输出到串中返回值: 正常情况下返回生成字串的长度(除去\0),错误情况返回负值用 法: int vsprintf(char *string, char *format, va_list param);// 将param 按格式format写入字符串string中注: 该函数会出现内存溢出情况,建议使用vsnprintf程序例:*/#include <stdarg.h>char buffer[80];int vspf(char *fmt, ...){va_list argptr;int cnt;va_start(argptr, fmt);cnt = vsprintf(buffer, fmt, argptr);va_end(argptr);return(cnt);}int main(void){int inumber = 30;float fnumber = 90.0;char string[4] = "abc";vspf("%d %f %s", inumber, fnumber, string);printf("%s\n", buffer);return 0;}该程序结果为30 90.000000 abc

知识点2:参数在堆栈中分布,位置

在进程中,堆栈地址是从高到低分配的.当执行一个函数的时候,将参数列表入栈,压入堆栈的高地址部分,然后入栈函数的返回地址,接着入栈函数的执行代码,这个入栈过程,堆栈地址不断递减,一些黑客就是在堆栈中修改函数返回地址,执行自己的代码来达到执行自己插入的代码段的目的.

总之,函数在堆栈中的分布情况是:地址从高到低,依次是:函数参数列表,函数返回地址,函数执行代码段.

堆栈中,各个函数的分布情况是倒序的.即最后一个参数在列表中地址最高部分,第一个参数在列表地址的最低部分.参数在堆栈中的分布情况如下:

最后一个参数

倒数第二个参数

...

第一个参数

函数返回地址


知识点3:堆栈分布及其设置

C/C++

一个由C/C++编译的程序占用的内存分为以下几个部分
1、栈区(stack)— 由编译器自动分配释放 ,存放函数的参数名,局部变量的名等。其操作方式类似于数据结构中的栈。
2、堆区(heap)— 由程序员分配释放, 若程序员不释放,程序结束时可能由OS回收。注意它与数据结构中的堆是两回事,分配方式倒是类似于链表。
3、全局区(静态区)(static)—全局变量和静态变量的存储是放在一块的,初始化的全局变量和静态变量在一块区域, 未初始化的全局变量和未初始化的静态变量在相邻的另一块区域。程序结束后由系统释放。
4、文字常量区—常量字符串就是放在这里的,程序结束后由系统释放 。
5、程序代码区— 存放函数体的二进制代码。

例子程序

这是一个前辈写的,非常详细
1
2
3
4
5
6
7
8
9
10
11
12
13
//main.cpp
inta = 0;//全局初始化区
char*p1;//全局未初始化区
main()
{
    intb;//栈
    chars[] = "abc";//栈
    char*p2;//栈
    char*p3 = "123456";//123456\0在常量区,p3在栈上。
    staticintc =0;//全局(静态)初始化区
    p1 = (char*)malloc(10);
    p2 = (char*)malloc(20);//分配得来的10和20字节的区域就在堆区。
}
strcpy(p1, "123456"); 123456\0放在常量区,编译器可能会将它与p3所指向的"123456"优化成一个地方。

小结

堆和栈的区别可以用如下的比喻来看出:
使用栈就象我们去饭馆里吃饭,只管点菜(发出申请)、付钱、和吃(使用),吃饱了就走,不必理会切菜、洗菜等准备工作和洗碗、刷锅等扫尾工作,他的好处是快捷,但是自由度小。
使用堆就象是自己动手做喜欢吃的菜肴,比较麻烦,但是比较符合自己的口味,而且自由度大。



1 0
原创粉丝点击