• <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>
            posts - 15,comments - 21,trackbacks - 0

            先看一個簡單的使用例子

            求任意個自然數的平方和:

            int SqSum(int n,)   
            {   
                va_list arg_ptr;   
                
            int sum = 0,_n = n;   
                arg_ptr 
            = va_start(arg_ptr,n);   
                
            while(_n != 0)   
                {   
                    sum 
            += (_n*_n);   
                    _n 
            = va_arg(arg_ptr,int);   
                }   
                va_end(arg_ptr);   
                
            return sum;   
            }
            首先解釋下函數參數入棧情況 在VC等絕大多數C編譯器中,默認情況下,參數進棧的順序是由右向左的,因此,參數進棧以后的內存模型如下圖所示:
            最后一個固定參數的地址位于第一個可變參數之下,并且是連續存儲的。
            | 最后一個可變參數(高內存地址處) | 第N個可變參數 | 第一個可變參數 | 最后一個固定參數 | 第一個固定參數(低內存地址處)
            明白上面那個順序,就知道其實可變參數就是玩弄參數的地址,已達到“不定”的目的 下面我摘自VC中的源碼來解釋
            va_list,va_start,va_arg,va_end宏
            1.其實va_list就是我們平時經常用的char* typedef char * va_list;
            2.va_start該宏的目的就是將指針指向最后一個固定參數的后面,即第一個不定參數的起始地址 #define va_start(ap,v)( ap = (va_list)&v + _INTSIZEOF(v) ) v即表示最后一個固定參數,&v表示v的地址, #define _INTSIZEOF(n) ( (sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1) ) 該宏其實是一個內存對齊的操作。即表示大于sizeof(n)且為sizeof(int)倍數的最小整數。這句話有點繞,其實舉幾個例子就簡單了。比如1--4,則返回4,5--8則返回8
            3.va_arg 該宏的目的是將ap指針繼續后移,讀取后面的參數,t表示參數類型。該宏首先將ap指針移動到下一個參數的起始地址ap += _INTSIZEOF(t),然后將本參數的值返回 #define va_arg(ap,t) ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )
            4.va_end將指針賦空 #define va_end(ap) ap = (va_list)0 有了這個分析我們可以把上例中的代碼重新翻譯下
            int SqSum(int n,)   
            {   
                
            char *arg_ptr;   
                
            int sum = 0,_n = n;   
                arg_ptr 
            = (char *)&+ 4;//本機上sizeof(int) = 4   
                while(_n != 0)   
                {   
                    sum 
            += (_n*_n);   
                    arg_ptr 
            += 4;   
                    _n 
            = *(int *)(arg_ptr-4);   
                }   
                arg_ptr 
            = (void*)0;   
            }  
            這樣我們也可以寫出我們自己的printf了
            posted on 2012-07-12 09:51 梨樹陽光 閱讀(1066) 評論(1)  編輯 收藏 引用 所屬分類: C

            FeedBack:
            # re: 不定參數的分析
            2012-07-31 14:18 | 叫我老王吧
            為什么是+4不是-4,固定參數在高地址,不是應該往下嗎?  回復  更多評論
              
            狠狠狠色丁香婷婷综合久久俺| 伊人久久成人成综合网222| 国产一级持黄大片99久久| jizzjizz国产精品久久| 久久久久国色AV免费看图片| 亚洲中文字幕无码久久综合网| 狠色狠色狠狠色综合久久| 久久久久久亚洲精品无码| 久久精品日日躁夜夜躁欧美| 久久亚洲欧美日本精品| 中文字幕久久精品无码| 久久久精品国产亚洲成人满18免费网站| 亚洲另类欧美综合久久图片区| 久久线看观看精品香蕉国产| 欧美一区二区久久精品| 久久99精品久久久久久噜噜| 潮喷大喷水系列无码久久精品| 久久婷婷午色综合夜啪| 久久精品国产99久久丝袜| 久久99国产精品二区不卡| 72种姿势欧美久久久久大黄蕉| 久久精品国产久精国产果冻传媒 | 亚洲午夜无码AV毛片久久| 国产成人久久激情91| 久久久国产精品亚洲一区 | 99久久无色码中文字幕| 中文字幕日本人妻久久久免费| 亚洲国产精品综合久久网络| 久久久精品人妻无码专区不卡| 久久se精品一区二区影院| 青青国产成人久久91网| 久久亚洲精品中文字幕三区| 久久久久久狠狠丁香| 国产一区二区三区久久精品| 久久久无码精品亚洲日韩蜜臀浪潮| 99久久国产精品免费一区二区| 久久九九久精品国产免费直播| 亚洲精品乱码久久久久久中文字幕| 亚洲午夜久久久影院伊人| 国产精品禁18久久久夂久| 欧美精品一本久久男人的天堂 |