• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            Prayer

            在一般中尋求卓越
            posts - 1256, comments - 190, trackbacks - 0, articles - 0
              C++博客 :: 首頁 :: 新隨筆 :: 聯(lián)系 :: 聚合  :: 管理

            va_start va_end va_arg是做什么用的

            Posted on 2009-01-13 11:10 Prayer 閱讀(623) 評論(0)  編輯 收藏 引用 所屬分類: C/C++
            ◎用法:
            func( Type para1, Type para2, Type para3, ... )
            {
            /****** Step 1 ******/
            va_list ap;
            va_start( ap, para3 ); //一定要“...”之前的那個參數(shù)

            /****** Step 2 ******/
            //此時ap指向第一個可變參數(shù)
            //調(diào)用va_arg取得里面的值

            Type xx = va_arg( ap, Type );

            //Type一定要相同,如:
            //char *p = va_arg( ap, char *);
            //int i = va_arg( ap, int );

            //如果有多個參數(shù)繼續(xù)調(diào)用va_arg

            /****** Step 3 ******/
            va_end(ap); //For robust!
            }

            ◎研究:
            typedef char * va_list;

            #define va_start _crt_va_start
            #define va_arg _crt_va_arg
            #define va_end _crt_va_end

            #define _crt_va_start(ap,v) ( ap = (va_list)_ADDRESSOF(v) + _INTSIZEOF(v) )
            #define _crt_va_arg(ap,t) ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )
            #define _crt_va_end(ap) ( ap = (va_list)0 )
            va_list argptr;
            C語言的函數(shù)是從右向左壓入堆棧的,調(diào)用va_start后,
            按定義的宏運算,_ADDRESSOF得到v所在的地址,然后這個
            地址加上v的大小,則使ap指向第一個可變參數(shù)如圖:

            棧底 高地址
            | .......
            | 函數(shù)返回地址
            | .......
            | 函數(shù)最后一個參數(shù)
            | ....
            | 函數(shù)第一個可變參數(shù) <--va_start后ap指向
            | 函數(shù)最后一個固定參數(shù)
            | 函數(shù)第一個固定參數(shù)
            棧頂 低地址


            然后,用va_arg()取得類型t的可變參數(shù)值, 先是讓ap指向下一個參數(shù):
            ap += _INTSIZEOF(t),然后在減去_INTSIZEOF(t),使得表達(dá)式結(jié)果為
            ap之前的值,即當(dāng)前需要得到的參數(shù)的地址,強制轉(zhuǎn)換成指向此參數(shù)的
            類型的指針,然后用*取值

            最后,用va_end(ap),給ap初始化,保持健壯性。

            example:(chenguiming)

            #include <stdio.h>
            #include <ctype.h>
            #include<stdlib.h>
            #include <stdarg.h>

            int average( int first, ... ) //變參數(shù)函數(shù),C++里也有
            {
            int count=0,i=first,sum=0;
            va_list maker; //va_list 類型數(shù)據(jù)可以保存函數(shù)的所有參數(shù),做為一個列表一樣保存
            va_start(maker,first); //設(shè)置列表的起始位置
            while(i!=-1)
            {
            sum+=i;
            count++;
            i=va_arg(maker,int);//返回maker列表的當(dāng)前值,并指向列表的下一個位置
            }
            return sum/count;

            }

            void main(void)
            {
            printf( "Average is: %d\n", average( 2, 3, 4,4, -1 ) );
            }
            日本久久久久亚洲中字幕| 色婷婷综合久久久久中文| 人妻精品久久久久中文字幕69 | 91精品国产91久久久久福利| 综合网日日天干夜夜久久 | 久久人人爽人人爽AV片| 久久久久国产一区二区三区| 久久天天躁狠狠躁夜夜2020一| 久久精品免费一区二区| AV无码久久久久不卡蜜桃| 中文字幕成人精品久久不卡| 久久久久香蕉视频| 久久偷看各类wc女厕嘘嘘| 久久99精品久久久久久| 国内精品九九久久精品| 久久99精品久久久久久不卡| 精品国产VA久久久久久久冰 | 欧美午夜A∨大片久久 | 久久久91人妻无码精品蜜桃HD| 婷婷久久香蕉五月综合加勒比| 国产亚洲成人久久| 性欧美丰满熟妇XXXX性久久久| 久久国产精品免费一区| 国产精品日韩欧美久久综合| 99蜜桃臀久久久欧美精品网站| 久久精品国产亚洲AV影院| 久久精品国产亚洲一区二区三区 | 久久精品国产精品亚洲| 午夜视频久久久久一区 | 久久乐国产综合亚洲精品| 亚洲国产成人久久笫一页| 久久天天婷婷五月俺也去| 久久99这里只有精品国产| 久久婷婷国产综合精品| 777久久精品一区二区三区无码| 香蕉久久影院| 国内精品伊人久久久久网站| 久久www免费人成看片| 久久96国产精品久久久| 久久夜色精品国产噜噜噜亚洲AV | 亚洲AV成人无码久久精品老人|