亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 學院 > 開發設計 > 正文

深入解析可變參數函數,以及可變參數常用到的宏函數解釋

2019-11-11 07:02:07
字體:
來源:轉載
供稿:網友

本文主要解釋大家經常遇到的可變參數函數的一些問題

函數的參數傳遞使用棧的方法進行傳值,所以第一個參數就在棧底,但是可變參數函數的困難就在于,不知道棧頂在哪,此時如果不做處理的話,程序就會很危險,這個需要開發者在函數中進行處理,例如常見的PRintf函數,就在第一個參數中指定了函數參數的個數,用%做參數標示。

下面舉個可變參數示例:

#include <stdarg.h> //使用va_list ,va_start等宏必須要包含的頭文件void arg_test(int i, char c ,...){ int j; va_list arg_ptr; //定義可變參數指針 va_start(arg_ptr, c); //設置可變參數指針指向...后的第一個參數地址//固定參數 printf("&i = %p i=%d /n", &i, i); //打印參數i的值以及在堆棧中的地址 printf("&c = %p c=%c /n", &c, c); //打印參數c的值以及在堆棧中的地址 //非固定參數 printf(" ----------下面是可變參數 ----------/n"); printf("arg_ptr = %p ", arg_ptr); //打印第一個可變參數地址 j = va_arg(arg_ptr, int); //獲取此時arg_ptr的值,并把arg_ptr后移到下一個參數地址 printf("j=%d /n", j); //打印參數j //判斷最后一個參數是否為-1,-1就結束 while (j != -1) { printf("arg_ptr = %p ", arg_ptr); //打印新的可變參數地址 j = va_arg(arg_ptr, int); //獲取此時arg_ptr的值,并把arg_ptr后移到下一個參數地址 printf("j=%d /n", j); //打印新的參數值 }}int main(int argc,char *argv[]){ int int_size = _INTSIZEOF(int); printf("int_size=%d/n", int_size); arg_test(1, 'c', 2,3,4,5,6,7,-1); return 0;}

說明: int int_size = _INTSIZEOF(int);得到int類型所占字節數 va_start(arg_ptr, i); 得到第一個可變參數地址 根據定義(va_list)&v得到起始參數的地址, 再加上_INTSIZEOF(v) ,就是其實參數下一個參數的地址,即函數void arg_test(int i, …)中參數 i 后的第一個可變參數地址. j=va_arg(arg_ptr, int); 得到arg_ptr指向的可變參數的值,并且arg_ptr指針上移一個_INTSIZEOF(int),即指向下一個可變參數的地址. va_end(arg_ptr);置空arg_ptr,即arg_ptr=(void *)0;

運算結果為: 這里寫圖片描述

關于可變參數的宏,再解釋一次,在VS中,聲明如下:

typedef char * va_list; #define _INTSIZEOF(n) ((sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1) ) #define va_start(ap,v) ( ap = (va_list)&v + _INTSIZEOF(v) ) #define va_arg(ap,t) ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) ) #define va_end(ap) ( ap = (va_list)0 )

網上有個帖子介紹這幾個宏說的不錯,如下: 1、首先把va_list被定義成char*,這是因為在我們目前所用的PC機上,字符指針類型可以用來存儲內存單元地址。而在有的機器上va_list是被定義成void*的 2、定義_INTSIZEOF(n)主要是為了某些需要內存的對齊的系統.這個宏的目的是為了得到最后一個固定參數的實際內存大小。在我的機器上直接用sizeof運算符來代替,對程序的運行結構也沒有影響。(后文將看到我自己的實現)。 3、va_start的定義為 &v+_INTSIZEOF(v) ,這里&v是最后一個固定參數的起始地址,再加上其實際占用大小后,就得到了第一個可變參數的起始內存地址。所以我們運行va_start(ap, v)以后,ap指向第一個可變參數在的內存地址,有了這個地址,以后的事情就簡單了。 這里要知道兩個事情: ⑴在intel+windows的機器上,函數棧的方向是向下的,棧頂指針的內存地址低于棧底指針,所以先進棧的數據是存放在內存的高地址處。 (2)在VC等絕大多數C編譯器中,默認情況下,參數進棧的順序是由右向左的,因此,參數進棧以后的內存模型如下圖所示:最后一個固定參數的地址位于第一個可變參數之下,并且是連續存儲的。 |————————–|

最后一個可變參數 ->高內存地址處
第N個可變參數 ->va_arg(arg_ptr,int)后arg_ptr所指的地方,
即第N個可變參數的地址。
第一個可變參數 ->va_start(arg_ptr,start)后arg_ptr所指的地方
即第一個可變參數的地址
最后一個固定參數 -> start的起始地址
……………..
-> 低內存地址處

(4) va_arg():有了va_start的良好基礎,我們取得了第一個可變參數的地址,在va_arg()里的任務就是根據指定的參數類型取得本參數的值,并且把指針調到下一個參數的起始地址。 因此,現在再來看va_arg()的實現就應該心中有數了: #define va_arg(ap,t) ( (t )((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) ) 這個宏做了兩個事情, ①用用戶輸入的類型名對參數地址進行強制類型轉換,得到用戶所需要的值 ②計算出本參數的實際大小,將指針調到本參數的結尾,也就是下一個參數的首地址,以便后續處理。 (5)va_end宏的解釋:x86平臺定義為ap=(char*)0;使ap不再 指向堆棧,而是跟NULL一樣.有些直接定義為((void*)0),這樣編譯器不會為va_end產生代碼,例如gcc在linux的x86平臺就是這樣定義的. 在這里大家要注意一個問題:由于參數的地址用于va_start宏,所以參數不能聲明為寄存器變量或作為函數或數組類型. 關于va_start, va_arg, va_end的描述就是這些了,我們要注意的 是不同的操作系統和硬件平臺的定義有些不同,但原理卻是相似的.

如果你對上面說的完全理解了,那么其實那些宏你也可以不用,如果你只是簡單使用,那么會使用宏就足夠了。 如下:

void Li_prinf(char * str, ...){ char * start = str; //定義字符指針 //char obj; //int leth; char* arg_ptr; //定義可變參數指針 arg_ptr = (char *)&str + 4; //設置可變指針指向為第一個可變參數 , //!??!注意這里需要取地址,因為參數進入堆棧的是指針,而不是具體的值?。?! // 這里為什么+4 因為參數指針大小就是4個字節,除非操作系統不同 while (*start) { if (*start == '%') { switch (*(++start)) { case 'c': //std::cout << va_arg(arg_ptr, char) << std::endl; //obj = (*(char*)((arg_ptr += ((sizeof(char) + sizeof(int) - 1) & ~(sizeof(int) - 1))) - ((sizeof(char) + sizeof(int) - 1) & ~(sizeof(int) - 1)))); //obj = (*(char*)((arg_ptr += 4) - 4)); //使用C++打印數值比較方面,當然用putchar也可以,只不過面對int、浮點值、字符串等需要特殊處理,有興趣的自己完善 std::cout << (*(char*)((arg_ptr += 4) - 4)) << std::endl; start++; break; case 'd': std::cout << (*(int*)((arg_ptr += 4) - 4)) << std::endl; start++; break; case 's': std::cout << (*(char**)((arg_ptr += 4) - 4)) << std::endl; start++; break; default: break; } } else { putchar(*start); start++; } } //設置指針指向0 arg_ptr = 0;}

這里就重寫了printf函數,有興趣的可以自己完善,很久沒寫博客了,如果寫的有誤,望各位大神告知


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
国产精品一区二区3区| 欧美性精品220| 91欧美精品午夜性色福利在线| 久久青草精品视频免费观看| 精品国内亚洲在观看18黄| 欧美一级免费看| 久久精品视频99| 欧美老少做受xxxx高潮| 日韩精品视频在线观看免费| 亚洲第一区第一页| 欧美精品一本久久男人的天堂| 欧美—级a级欧美特级ar全黄| 亚洲欧洲中文天堂| 日韩性xxxx爱| 国产日本欧美一区二区三区| 亚洲一区精品电影| 中文字幕精品国产| 91成人天堂久久成人| 精品国产鲁一鲁一区二区张丽| 九九热r在线视频精品| 国产网站欧美日韩免费精品在线观看| 国内精品视频久久| 亚洲国产精品一区二区久| 日韩av免费网站| 91视频九色网站| 成人精品视频99在线观看免费| 日韩av在线免费观看| 欧美精品日韩三级| 97人人做人人爱| 伊是香蕉大人久久| 欧美激情精品久久久久久免费印度| 亚洲精品久久久久久久久久久久| 亚洲天堂成人在线视频| 亚洲a级在线播放观看| 8050国产精品久久久久久| 国产精品久久久久久久久免费| 欧美在线精品免播放器视频| 性欧美暴力猛交69hd| 亚洲国产婷婷香蕉久久久久久| 永久免费毛片在线播放不卡| 亚洲最新av在线网站| 欧美激情精品久久久久久大尺度| 国语自产精品视频在免费| 亚洲护士老师的毛茸茸最新章节| 国产欧美日韩精品丝袜高跟鞋| 在线激情影院一区| 国产+成+人+亚洲欧洲| 2018国产精品视频| 777精品视频| 亚洲图中文字幕| 精品国产91久久久久久老师| 亚洲伊人久久大香线蕉av| 欧美激情国产日韩精品一区18| 亚洲深夜福利网站| 久久99久久99精品免观看粉嫩| 欧美日韩美女在线| 国产精品jizz在线观看麻豆| 国产福利成人在线| 中文字幕不卡av| 精品国内自产拍在线观看| 77777亚洲午夜久久多人| 久久久精品日本| 国产在线日韩在线| 精品国偷自产在线视频| 欧美日韩免费在线观看| 成人精品aaaa网站| 一区国产精品视频| 91麻豆国产语对白在线观看| 久久人人爽人人爽人人片亚洲| 国产日韩在线一区| 亚洲精品suv精品一区二区| 色香阁99久久精品久久久| 国产成人一区二区三区电影| 久久久久一本一区二区青青蜜月| 992tv在线成人免费观看| 亚洲精品国产品国语在线| 最新69国产成人精品视频免费| 久久成人综合视频| 91av在线影院| 欧美国产乱视频| 亚洲精品大尺度| 国产亚洲精品成人av久久ww| 91最新国产视频| 久久久久亚洲精品成人网小说| 欧美极品少妇xxxxⅹ免费视频| 国产欧美在线观看| 欧美电影在线观看| 日日摸夜夜添一区| 日韩av网站大全| 欧美裸体xxxx极品少妇软件| 成人av色在线观看| 久久精品电影网| 国产精品久久久久久久久久小说| 国产精品福利在线观看网址| 精品亚洲一区二区三区在线播放| 国产成人精品久久久| 精品成人av一区| 亚洲国产精品大全| 国产香蕉精品视频一区二区三区| 91av免费观看91av精品在线| 亚洲国产精品久久精品怡红院| 日韩黄色高清视频| 日韩欧美在线一区| 亚洲国产精品悠悠久久琪琪| 国产精品久久久一区| 97国产真实伦对白精彩视频8| 精品亚洲国产视频| 91黑丝高跟在线| 国产午夜精品视频| 欧美贵妇videos办公室| 欧美大学生性色视频| 国产精品久久久久久亚洲影视| 国内免费久久久久久久久久久| 在线电影欧美日韩一区二区私密| 北条麻妃99精品青青久久| 欧美黑人又粗大| 91夜夜未满十八勿入爽爽影院| 欧美亚洲一级片| 日韩一二三在线视频播| 国产精品海角社区在线观看| 国产91在线播放九色快色| 91欧美精品成人综合在线观看| 2019精品视频| 日韩电影网在线| 欧美区二区三区| 国产精品视频公开费视频| 欧美大尺度激情区在线播放| 超在线视频97| 国产日韩综合一区二区性色av| 亚洲国产一区二区三区四区| 亚洲高清免费观看高清完整版| 性色av一区二区三区在线观看| 久久精品国亚洲| 国产999精品久久久| 日本精品视频在线观看| 日韩一区二区三区在线播放| 国产亚洲激情视频在线| 在线视频国产日韩| 日韩美女在线观看| 亚洲成人av在线播放| 久久久女人电视剧免费播放下载| 2019亚洲男人天堂| 久久久久久亚洲精品不卡| 国产精品中文在线| 欧美大片免费观看在线观看网站推荐| 亚洲人成自拍网站| 亚洲男人av在线| 青青草原一区二区| 亚洲人成电影网| 久久99亚洲精品| 欧美成人精品激情在线观看| 久久精品91久久香蕉加勒比| 欧美尺度大的性做爰视频| 97碰碰碰免费色视频| 国产精品高清在线| 日本最新高清不卡中文字幕| 欧美日韩精品国产| 亚洲一区免费网站| 中文字幕自拍vr一区二区三区| 91久久精品国产91久久性色| 夜色77av精品影院| 色多多国产成人永久免费网站| 国产精品99一区| 伊人久久久久久久久久久久久|