青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

Kisser Leon

這個(gè)kisser不太冷
posts - 100, comments - 102, trackbacks - 0, articles - 0

可變參數(shù)研究

Posted on 2007-04-10 10:38 kk 閱讀(676) 評(píng)論(0)  編輯 收藏 引用 所屬分類: IT

一. 何謂可變參數(shù)
int printf( const char* format, ...);
這是使用過(guò)C語(yǔ)言的人所再熟悉不過(guò)的printf函數(shù)原型,它的參數(shù)中就有固定參數(shù)format和可變參數(shù)(用”…”表示). 而我們又可以用各種方式來(lái)調(diào)用printf,:
printf("%d",value);
printf("%s",str);
printf("the number is %d ,string is:%s", value, str);
.實(shí)現(xiàn)原理
C 語(yǔ)言用宏來(lái)處理這些可變參數(shù)。這些宏看起來(lái)很復(fù)雜,其實(shí)原理挺簡(jiǎn)單,就是根據(jù)參數(shù)入棧的特點(diǎn)從最靠近第一個(gè)可變參數(shù)的固定參數(shù)開始,依次獲取每個(gè)可變參數(shù) 的地址。下面我們來(lái)分析這些宏。在VC中的stdarg.h頭文件中,針對(duì)不同平臺(tái)有不同的宏定義,我們選取X86平臺(tái)下的宏定義:
typedef char *va_list;
/*va_list被定義成char*,這是因?yàn)樵谖覀兡壳八玫?/span>PC機(jī)上,字符指針類型可以用來(lái)存儲(chǔ)內(nèi)存單元地址。而在有的機(jī)器上va_list是被定義成void**/
#define _INTSIZEOF(n) ( (sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1) )
/*_INTSIZEOF (n)宏是為了考慮那些內(nèi)存地址需要對(duì)齊的系統(tǒng),從宏的名字來(lái)應(yīng)該是跟sizeof(int)對(duì)齊。一般的sizeof(int)=4,也就是參數(shù)在內(nèi)存 中的地址都為4的倍數(shù)。比如,如果sizeof(n)14之間,那么_INTSIZEOF(n)4;如果sizeof(n)58之間,那么 _INTSIZEOF(n)=8*/
#define va_start(ap,v)( ap = (va_list)&v + _INTSIZEOF(v) )
/*va_start 的定義為 &v+_INTSIZEOF(v) ,這里&v是最后一個(gè)固定參數(shù)的起始地址,再加上其實(shí)際占用大小后,就得到了第一個(gè)可變參數(shù)的起始內(nèi)存地址。所以我們運(yùn)行va_start (ap, v)以后,ap指向第一個(gè)可變參數(shù)在的內(nèi)存地址*/
#define va_arg(ap,t) ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )
/*這個(gè)宏做了兩個(gè)事情,
①用用戶輸入的類型名對(duì)參數(shù)地址進(jìn)行強(qiáng)制類型轉(zhuǎn)換,得到用戶所需要的值
②計(jì)算出本參數(shù)的實(shí)際大小,將指針調(diào)到本參數(shù)的結(jié)尾,也就是下一個(gè)參數(shù)的首地址,以便后續(xù)處理。*/
  #define va_end(ap) ( ap = (va_list)0 )
/*x86 平臺(tái)定義為ap=(char*)0;使ap不再 指向堆棧,而是跟NULL一樣.有些直接定義為((void*)0),這樣編譯器不會(huì)為va_end產(chǎn)生代碼,例如gcclinuxx86平臺(tái)就是這 樣定義的. 在這里大家要注意一個(gè)問題:由于參數(shù)的地址用于va_start,所以參數(shù)不能聲明為寄存器變量或作為函數(shù)或數(shù)組類型. */

以下再用圖來(lái)表示:

VC等絕大多數(shù)C編譯器中,默認(rèn)情況下,參數(shù)進(jìn)棧的順序是由右向左的,因此,參數(shù)進(jìn)棧以后的內(nèi)存模型如下圖所示:最后一個(gè)固定參數(shù)的地址位于第一個(gè)可變參數(shù)之下,并且是連續(xù)存儲(chǔ)的。
|——————————————————————————|
|最后一個(gè)可變參數(shù) | ->高內(nèi)存地址處
|——————————————————————————|
...................
|——————————————————————————|
|N個(gè)可變參數(shù) | ->va_arg(arg_ptr,int)arg_ptr所指的地方,
| | 即第N個(gè)可變參數(shù)的地址。
|——————————————— |
………………………….
|——————————————————————————|
|第一個(gè)可變參數(shù) | ->va_start(arg_ptr,start)arg_ptr所指的地方
| | 即第一個(gè)可變參數(shù)的地址
|——————————————— |
|———————————————————————— ——|
| |
|最后一個(gè)固定參數(shù) | -> start的起始地址
|—————————————— —| .................
|—————————————————————————— |
| |
|——————————————— |-> 低內(nèi)存地址處

.printf研究

下面是一個(gè)簡(jiǎn)單的printf函數(shù)的實(shí)現(xiàn),參考了中的156頁(yè)的例子,讀者可以結(jié)合書上的代碼與本文參照。
#include "stdio.h"
#include "stdlib.h"
void myprintf(char* fmt, ...) //一個(gè)簡(jiǎn)單的類似于printf的實(shí)現(xiàn),//參數(shù)必須都是int 類型
{
char* pArg=NULL; //等價(jià)于原來(lái)的va_list
char c;

pArg = (char*) &fmt; //注意不要寫成p = fmt !!因?yàn)檫@里要對(duì)//參數(shù)取址,而不是取值
pArg += sizeof(fmt); //等價(jià)于原來(lái)的va_start

do
{
c =*fmt;
if (c != '%')
{
putchar(c); //照原樣輸出字符
}
else
{
//按格式字符輸出數(shù)據(jù)
switch(*++fmt)
{
case 'd':
printf("%d",*((int*)pArg));
break;
case 'x':
printf("%#x",*((int*)pArg));
break;
default:
break;
}
pArg += sizeof(int); //等價(jià)于原來(lái)的va_arg
}
++fmt;
}while (*fmt != '\0');
pArg = NULL; //等價(jià)于va_end
return;
}
int main(int argc, char* argv[])
{
int i = 1234;
int j = 5678;

myprintf("the first test:i=%d",i,j);
myprintf("the secend test:i=%d; %x;j=%d;",i,0xabcd,j);
system("pause");
return 0;
}
intel+win2k+vc6的機(jī)器執(zhí)行結(jié)果如下:
the first test:i=1234
the secend test:i=1234; 0xabcd;j=5678;

.應(yīng)用
求最大值:
#include //不定數(shù)目參數(shù)需要的宏
int max(int n,int num,...)
{
va_list x;//說(shuō)明變量x
va_start(x,num);//x被初始化為指向num后的第一個(gè)參數(shù)
int m=num;
for(int i=1;i {
//將變量x所指向的int類型的值賦給y,同時(shí)使x指向下一個(gè)參數(shù)
int y=va_arg(x,int);
if(y>m)m=y;
}
va_end(x);//清除變量x
return m;
}
main()
{
printf("%d,%d",max(3,5,56),max(6,0,4,32,45,533));
}

 

本文轉(zhuǎn)載自網(wǎng)上,本來(lái)是要注明出處的,結(jié)果別人也都是轉(zhuǎn)載的,呵呵。不過(guò)此文講的很不錯(cuò),很清楚,特別是把可變參數(shù)實(shí)現(xiàn)的那幾個(gè)宏!偶也是沖著這幾個(gè)宏去的。

#define _INTSIZEOF(n) ( (sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1) )
這個(gè)宏,一開始我就是想不明白!不知道是老了,還是笨了,或者是生銹了。想了好一會(huì)還是沒搞明白,不過(guò)看了一下本文的分析,一下子就明白了,那是相當(dāng)?shù)幕腥淮笪虬 ?/span>

#define va_arg(ap,t) ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )

這個(gè)宏,一開始也是沒有明白。多看了好幾遍才發(fā)現(xiàn)了奧秘所在啊,為什么要加一個(gè),然后再減一個(gè)呢?因?yàn)榈谝粋€(gè)加直接加到ap上去了,而后一個(gè)減只是減了一下括號(hào)內(nèi)的值,也就是當(dāng)前值了。

宏真是厲害啊!或者說(shuō)它應(yīng)用真廣!這不禁讓我想起以前看過(guò)的宏,怎么判斷是win還是linux平臺(tái)的,怎么判斷是32位的還是64位的。宏是一門學(xué)問啊。

本文的另一大優(yōu)點(diǎn)是,有非常簡(jiǎn)單的例子,看了就懂。恩。看了保管你就會(huì)用了。不過(guò)這年頭指針也是個(gè)好東西啊,需要什么,傳個(gè)指針就是傳了一切想要的東西啊,只要讓指針指向你需要的東西,可以是任意多的參數(shù)(不過(guò)這樣子的話,具體到哪個(gè)參數(shù)結(jié)束就要我們自己來(lái)定了,不像這里,所有的參數(shù)都已經(jīng)壓棧了,編譯器可以幫我們決定具體有多少個(gè)參數(shù),到什么時(shí)候結(jié)束)。

Have fun.

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            欧美a级大片| 在线免费一区三区| 制服丝袜亚洲播放| 欧美在线视频观看免费网站| 国产亚洲一区二区精品| 国产精品制服诱惑| 欧美大片免费久久精品三p | 亚洲国产精品激情在线观看| 欧美色视频日本高清在线观看| 久久婷婷久久一区二区三区| 欧美有码视频| 久久久噜噜噜久久中文字免| 欧美中文在线观看国产| 久久成人免费视频| 久久精品99无色码中文字幕| 久久免费精品日本久久中文字幕| 亚洲自拍另类| 久久久久国产精品一区| 久久久久久97三级| 免费成人黄色| 国产精品久久久久影院色老大| 欧美成人一区二区三区在线观看| 久久中文字幕一区| 欧美日韩一级视频| 国产精品亚洲аv天堂网| 午夜精品久久久久久99热软件| 欧美激情亚洲视频| 欧美电影免费观看高清完整版| 欧美福利在线观看| 亚洲精品网址在线观看| 欧美亚洲网站| 欧美激情亚洲综合一区| 国产精品国产三级国产普通话三级 | 欧美自拍偷拍| 一区二区三区日韩精品| 欧美一区二区大片| 国产精品久久久久久久电影| 亚洲国产成人不卡| 亚洲一区网站| 裸体女人亚洲精品一区| 亚洲午夜av在线| 欧美激情一区二区| 在线播放中文一区| 美日韩精品视频| 久久精品二区三区| 国产综合色产在线精品| 欧美一区二区三区免费视| 日韩视频在线播放| 国产精品国产三级国产aⅴ入口| 在线成人激情视频| 欧美成人高清视频| 欧美成人午夜激情在线| 日韩视频在线观看| 91久久香蕉国产日韩欧美9色| 久久精品国产亚洲一区二区三区| 国产精品欧美日韩久久| 欧美亚洲日本国产| 久久国产精品免费一区| 亚洲福利在线观看| 亚洲人成在线观看| 欧美日韩123| 亚洲欧美自拍偷拍| 久久久久国产精品人| 欧美在线观看你懂的| 国产日韩欧美在线看| 毛片精品免费在线观看| 欧美日韩免费观看一区| 欧美在线播放一区| 免费观看日韩av| 久久精品国产综合| 欧美日韩成人免费| 久久精品免视看| 欧美三区在线| 亚洲福利在线视频| 欧美有码在线视频| 亚洲一区二区三区四区中文| 欧美伊人久久| 欧美在线免费视频| 欧美日韩专区在线| 亚洲激情女人| 亚洲三级免费| 亚洲激情欧美| 国产精品99久久久久久久vr| 久久一区二区三区超碰国产精品| 久久爱www| 国产午夜精品麻豆| 亚洲女人小视频在线观看| 亚洲国产精品www| 亚洲欧洲一级| 亚洲淫性视频| 亚洲高清在线观看| 亚洲欧美精品在线| 国产亚洲福利社区一区| 性久久久久久久久| 亚洲人成毛片在线播放| 中文在线不卡视频| 在线播放亚洲| 国产精品超碰97尤物18| 亚洲欧美日韩在线综合| 亚洲三级视频在线观看| 国产女优一区| 国产精品大全| 欧美日韩国产首页在线观看| 欧美1区3d| 欧美亚洲第一区| 国产有码一区二区| 国产日韩欧美精品一区| 欧美成人免费大片| 亚洲专区欧美专区| 欧美巨乳波霸| 亚洲一品av免费观看| 欧美一区不卡| 欧美xart系列在线观看| 亚洲国产老妈| 亚洲欧美韩国| 亚洲国产精品99久久久久久久久| 亚洲精品之草原avav久久| 中国成人在线视频| 亚洲精品一二三区| 久久久久www| 老色批av在线精品| 亚洲深夜福利视频| 欧美日韩 国产精品| 欧美大片在线看| 国产一区二区丝袜高跟鞋图片| 亚洲第一精品影视| 国产欧美精品va在线观看| 久久成人18免费网站| 久久夜色精品国产亚洲aⅴ | 亚洲三级免费| 亚洲国产精品视频| 免费在线亚洲欧美| 亚洲国产精品久久人人爱蜜臀 | 免费亚洲电影在线观看| 亚洲视频在线观看| 激情久久婷婷| 久久国产精品久久久久久| 欧美激情91| 久久久综合香蕉尹人综合网| 欧美电影免费网站| 欧美大学生性色视频| 亚洲午夜av| 亚洲女人天堂av| 欧美一区永久视频免费观看| 亚洲午夜免费视频| 99www免费人成精品| 亚洲第一中文字幕| 欧美—级a级欧美特级ar全黄| 一本一本久久| 欧美激情精品久久久久久黑人 | 久久久久久久久一区二区| 亚洲一区自拍| 亚洲天堂av电影| 最新亚洲一区| 这里只有精品视频在线| 久久国产手机看片| 国产人久久人人人人爽| 欧美在线看片a免费观看| 久久久久天天天天| 亚洲第一福利在线观看| 欧美激情免费观看| 一本色道久久综合| 香蕉久久夜色精品国产| 国产一区在线免费观看| 美日韩丰满少妇在线观看| 亚洲国产另类 国产精品国产免费| 999在线观看精品免费不卡网站| 欧美日韩亚洲成人| 久久精品99国产精品日本| 亚洲国产精品一区二区久| 欧美精品在线观看| 亚洲午夜性刺激影院| 久久中文久久字幕| 一区二区精品在线观看| 国产亚洲毛片在线| 欧美精品久久一区| 亚洲欧美日韩成人高清在线一区| 久久视频免费观看| av成人免费观看| 国内一区二区在线视频观看| 欧美破处大片在线视频| 午夜性色一区二区三区免费视频| 欧美电影免费观看高清完整版| 久久福利精品| 国产视频一区在线观看| 久久综合网色—综合色88| 亚洲国产精品成人综合| 午夜精品短视频| 亚洲乱码国产乱码精品精天堂 | 欧美成人久久| 午夜精品久久| 日韩小视频在线观看专区| 国产自产在线视频一区| 国产精品初高中精品久久| 欧美wwwwww| 鲁鲁狠狠狠7777一区二区| 亚洲午夜久久久久久尤物| 亚洲精品1区2区| 美女视频一区免费观看| 久久av红桃一区二区小说|