zoukankan      html  css  js  c++  java
  • 深入C语言可变参数(va_arg,va_list,va_start,va_end,_INTSIZEOF)

    一、什么是可变参数

             在C语言编程中有时会遇到一些参数个数可变的函数,例如printf(),scanf()函数,其函数原型为:
    int printf(const char* format,…),int scanf(const char *format,…);它除了有一个参数format固定以外,后面跟着的参数的个数和类型是可变的(用三个点“…”做参数占位符),实际调用时可以有以下的形式:
               printf(“%d”,i);  printf(“%d,%c”,i,j);

    二、C语言---简单的可变参数例子(实现思想)

            1、整型数据的输出

    #include <stdio.h>
    #include <stdarg.h>
    void ar_cnt(int cnt,...);
    void ar_cst(char const *s,...);
    int main(int argc, char* argv[])
    {
        int in_size =_INTSIZEOF(int);
        printf("int_size=%d
    ",in_size);
        ar_cnt(5,1,2,3,4);
        return 0; 
    }
    void ar_cnt(int cnt,...)
    {
        int value1=0;
        int i=0;
        int arg_cnt = cnt;
        va_list arg_ptr;
        va_start(arg_ptr,cnt);
        for(i=0;i<cnt;i++)
        {
            value1=va_arg(arg_ptr,int);
            printf("posation %d=%d
    ",value1,i+1);
        }
        va_end(arg_ptr);
    }

    运行结果:

    image

            2、字符串的输出

    #include <stdio.h>
    #include <stdarg.h>
    
    void PrintLines(char *first,...)
    {
        char *str;
        va_list v1;
        str = first;
        va_start(v1,first);
        do 
        {
            printf("%s
    ",str);
            str=va_arg(v1,char*);
        } while (str != NULL );
        va_end(v1);
    }
    
    int main(int argc, char* argv[])
    {
        PrintLines("First","Second","Third","Fourth",NULL);
        return 0;
    }

    运行结果:

    image

               3、找出最大数

    #include <stdio.h>
    #include <stdarg.h>
    int FindMax(int amount,...)
    {
        int i,val,great;
        va_list v1;
        va_start(v1,amount);
        great=va_arg(v1,int);
        for(i=1;i<amount;i++)
        {
            val=va_arg(v1,int);
            great=(great>val)?great:val;
        }
        va_end(v1);
        return great;
    }
    
    int main(int argc, char* argv[])
    {
        int max=FindMax(5,100,20,456,102,4,300);
        printf("The Max one is %d
    ",max);
        return 0;
    }

    运行结果:

    image

    三、对va_arg,va_list,va_start,va_end,_INTSIZEOF剖析

    • 内存对齐  #define _INTSIZEOF(n) ((sizeof(n)+sizeof(int)-1)&~(sizeof(int) - 1) )
                   具体请参考:http://blog.csdn.net/swell624/article/details/3210779
    • 可变参数用到以下宏函数 
          <1>原型:
      void va_start(va_list arg_ptr,prev_param);

                  功能:以固定参数的地址为起点确定变参的内存起始地址,获取第一个参数的首地址

                  返回值:无

            <2>原型:va_list 类型的变量,va_list arg_ptr ,这个变量是指向参数地址的指针,因为得到参数的地址之后,再结合参数的类型,才能得到参数的值。

            <3>原型:type va_arg(va_list arg_ptr,type);

                  功能:获取下一个参数的地址

                返回值:根据传入参数类型决定返回值类型

            <4>原型:void  va_end(va_list arg_ptr);

                  功能:将arg_ptr指针置0

                 返回值:无

    • 使用可变参数应该有以下步骤

         ⑴在程序中将用到以下这些宏:

        void va_start( va_list arg_ptr, prev_param );

        type va_arg( va_list arg_ptr, type );

        void va_end( va_list arg_ptr );

        va在这里是variable-argument(可变参数)的意思.

        这些宏定义在stdarg.h中,所以用到可变参数的程序应该包含这个头文件.

        ⑵函数里首先定义一个va_list型的变量,这里是arg_ptr,这个变量是指向参数地址的指针.因为得到参数的地址之后,再结合参数的类型,才能得到参数的值。

        ⑶然后用va_start宏初始化⑵中定义的变量arg_ptr,这个宏的第二个参数是可变参数列表的前一个参数,也就是最后一个固定参数。

        ⑷然后依次用va_arg宏使arg_ptr返回可变参数的地址,得到这个地址之后,结合参数的类型,就可以得到参数的值。然后进行输出。

        ⑸设定结束条件,这里的条件就是判断参数值是否为-1。注意被调的函数在调用时是不知道可变参数的正确数目的,程序员必须自己在代码中指明结束条件。至于为什么它不会知道参数的数目,读者在看完下面这几个宏的内部实现机制后,自然就会明白。

    三、参考文献:
           http://www.chineselinuxuniversity.net/articles/26262.shtml

            http://www.cnblogs.com/wangyonghui/archive/2010/07/12/1776068.html

            http://www.2cto.com/kf/201204/129038.html

     

  • 相关阅读:
    java DMO及增删改查代码的自动生成
    如果有一天苹果免费了,支付宝怎么办
    jquery实现页面交互的几个小例子
    android中的所谓观察者模式
    随笔——runnable勘误以及其他
    android 源码角度全方位理解filter
    android 你的onfocuschangelistener和android:state_hovered为何不起作用
    android 如何阻断seekbar的触摸事件
    java 传值和传引用
    Android内存优化(使用SparseArray和ArrayMap代替HashMap)
  • 原文地址:https://www.cnblogs.com/haoyuanyuan/p/3221463.html
Copyright © 2011-2022 走看看