C語言可變參數(shù)列表的用法與深度剖析
前言
可變參數(shù)列表,使用起來像是數(shù)組,學(xué)習(xí)過函數(shù)棧幀的話可以發(fā)現(xiàn)實(shí)際上他也就是在棧區(qū)定義的一塊空間當(dāng)中連續(xù)訪問,不過他不支持直接在中間部分訪問。
聲明: 以下所有測試都是在x86,vs2013下完成的。
一、可變參數(shù)列表是什么?
在我們初始C語言的第一節(jié)課的時(shí)候我們就已經(jīng)接觸了可變參數(shù)列表,在printf的過程當(dāng)中我們通??梢詡鬟f大量要打印的參數(shù),但是我們卻不知道他是如何做到的,今天就帶大家剖析它。
二、怎么用可變參數(shù)列表
首先我們要引入windows.h的頭文件
然后我們先要介紹以下幾個(gè)宏。在這里我們先簡述它的功能,在后面會有詳細(xì)的講解,這里是為了方便大家入門。
typedef char* va_list; //類型的重定義 #define _ADDRESSOF(v) (&(v))//一個(gè)取地址的宏。
1._ADDRESSOF:取傳入變量的地址。
#define _INTSIZEOF(n) \ ((sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1))
2._INTSIZEOF:該宏功能是讓n的類型往4的倍數(shù)上取整。
#define _INTSIZEOF(n)\ ( (sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1) )
下面一段代碼進(jìn)行解釋:
#pragma pack(1)//設(shè)置默認(rèn)對其數(shù)為1 struct A { char ch[11]; }; int main() { printf("int : %d\n", _INTSIZEOF(int)); printf("double: %d\n", _INTSIZEOF(double)); printf("short: %d\n", _INTSIZEOF(short)); printf("float: %d\n", _INTSIZEOF(float)); printf("long long int: %d\n", _INTSIZEOF(long long int)); printf("struct A:%d\n", _INTSIZEOF(struct A)); return 0; }
結(jié)果:
3.__crt_va_start_a:取變量v的地址強(qiáng)轉(zhuǎn)為char*然后向指向v類型對其數(shù)后,即找到第一個(gè)可變參數(shù)列表當(dāng)中的變量!
#define __crt_va_start_a(ap, v) \ ((void)(ap = (va_list)_ADDRESSOF(v) + _INTSIZEOF(v)))
4.__crt_va_arg:將ap提前指向下一個(gè)要訪問的位置,并且返回當(dāng)前訪問的內(nèi)容。 注意+=后ap指向下一個(gè)要訪問的地址,但是返回的內(nèi)容是當(dāng)前的。
#define __crt_va_arg(ap, t) \ (*(t*)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)))
5.__crt_va_end:將ap置成NULL。
#define __crt_va_end(ap)\ ((void)(ap = (va_list)0))
緊接著我們看一下以下幾個(gè)定義。
#define va_start __crt_va_start #define va_arg __crt_va_arg #define va_end __crt_va_end
測試:找一組不存放在數(shù)組當(dāng)中的最大的一個(gè)數(shù)據(jù)返回。
int Find_Max(int num, ...) { //定義一個(gè)char* 的變量arg va_list arg; //將arg指向第一個(gè)可變參數(shù) va_start(arg, num); //將max置成第一個(gè)可變參數(shù),然后arg指向下一個(gè)可變參數(shù) int max = va_arg(arg, int); //循環(huán)num-1次,訪問完剩下的可變參,找到最大的賦值給max for (int i = 1; i < num; ++i) { int r; if (max < (r = va_arg(arg, int))) { max = r; } } //將arg指針變量置成NULL,避免野指針 va_end(arg); return max; }
int main() { int ret = Find_Max(5, 0x1, 0x2, 0x3, 0x4, 0x5); printf("ret :%d\n", ret); return 0; }
結(jié)果:
三、對于宏的深度剖析
雖然在Linux下的進(jìn)程地址空間是由高到低排列的,但是由于vs下的內(nèi)存是從低字節(jié)到高字節(jié)的,我們的棧會和linux下畫的不太一樣,但是都是朝著低地址方向擴(kuò)展的。這是方便大家理解。
Linux的進(jìn)程地址空間示意圖:
代碼棧幀示意圖:
隱式類型轉(zhuǎn)換
舉個(gè)栗子,當(dāng)我們執(zhí)行下面的代碼,當(dāng)我們以char類型傳參,但函數(shù)體依舊以int的步長獲取,此時(shí)會出錯(cuò)嗎?
#include<stdio.h> #include<windows.h> int Find_Max(int num, ...) { //定義一個(gè)char* 的變量arg va_list arg; //將arg指向第一個(gè)可變參數(shù) va_start(arg, num); //將max置成第一個(gè)可變參數(shù),然后arg指向下一個(gè)可變參數(shù) int max = va_arg(arg, int); //循環(huán)num-1次,訪問完剩下的可變參,找到最大的賦值給max for (int i = 1; i < num; ++i) { int r; if (max < (r = va_arg(arg, int))) { max = r; } } //將arg指針變量置成NULL,避免野指針 va_end(arg); return max; } int main() { char a = '1'; //ascii值: 49 char b = '2'; //ascii值: 50 char c = '3'; //ascii值: 51 char d = '4'; //ascii值: 52 char e = '5'; //ascii值: 53 int ret = Find_Max(5, a, b, c, d, e); //int ret = Find_Max(5, 0x1, 0x2, 0x3, 0x4, 0x5); printf("ret :%d\n", ret); system("pause"); }
答案:
不會的,由于壓棧的時(shí)候是通過寄存器傳參的,32位下的寄存器就是4個(gè)字節(jié)。
壓棧時(shí)的匯編:其中第一條不是mov,而是movsx,即匯編語言數(shù)據(jù)傳送指令MOV的變體。帶符號擴(kuò)展,并傳送。也就是整形提升。
同理:用float傳參,用double字長走,也是沒有問題的。
總結(jié):
所以我們習(xí)慣在函數(shù)體內(nèi)部(Find_Max)用int/double為長度走,而傳參的時(shí)候我們可以用char/short/float等等類型。
注意:
64位下的定義和32位下差異是很大的。
為什么按照4字節(jié)對齊:
先前講到在短整型,在壓棧的過程中會發(fā)生整形提升,那么從棧幀中要拿到對應(yīng)的數(shù)據(jù)也要按照對應(yīng)的方法提取。
_INTSIZEOF的數(shù)學(xué)理解:
_INTSIZEOF(n)的意思:計(jì)算一個(gè)最小數(shù)字x,滿足 x>=n && x%4==0,n表示sizeof(n)的值。即該類型的大小要滿足往n的整數(shù)倍對齊,且最小不能小于n。
以4字節(jié)對齊為栗子:
n%4 == 0,則 ret = n;
n %4 != 0 , 則 ret = (n+ 4 - 1)/4 *4;
(n+ 4 - 1)/4 -->假設(shè) n為1到4,那么(n + 4 - 1)/4的結(jié)果都是1,再乘上對其數(shù)4就是以4對齊的最小對齊數(shù)了。就能將這4個(gè)數(shù)值范圍
的最小對齊倍數(shù)
控制在同一個(gè)值。
我們觀察(n+ 4 -1)/4 *4,/4實(shí)際上就是將二進(jìn)制序列往右移,*4就是把二進(jìn)制序列往左移動,這一來一回實(shí)際上就是把最低兩位置成0,那么我們還可以簡化成:
(n+ 4 -1) & ~3 ,也就是源碼當(dāng)中的定義了!!
#define _INTSIZEOF(n)\ ( (sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1) )
對兩個(gè)函數(shù)的重新認(rèn)知
對printf的理解:
在上述的例子中,宏是無法判斷實(shí)際存在參數(shù)的數(shù)量,以及實(shí)際參數(shù)的類型的,那么在printf當(dāng)中,必定有能夠確定參數(shù)數(shù)量以及辨別參數(shù)類型的方法,實(shí)際上也就是%c,%d,%lf,其中%的數(shù)量除了%%外的%的數(shù)量實(shí)際上就能讓我們得知參數(shù)的數(shù)量,而%c,%d,實(shí)際上也就說明了對應(yīng)的類型。
對exec系列的理解:
在進(jìn)程控制,當(dāng)時(shí)講述了實(shí)際上只有一個(gè)系統(tǒng)調(diào)用execve,其他函數(shù)exec函數(shù)最終都是要調(diào)用execve函數(shù),那么是如何實(shí)現(xiàn)從參數(shù)l到v這個(gè)過程的呢?
答案:
實(shí)際上訪問到null為止,傳參的數(shù)量用一個(gè)count一直計(jì)數(shù)就能拿到,而類型毫無疑問就是char*,我們可以用strlen去計(jì)算要走多長。(不過兩個(gè)char數(shù)組通常會間隔多8個(gè)字節(jié))
總結(jié)
到此這篇關(guān)于C語言可變參數(shù)列表的文章就介紹到這了,更多相關(guān)C語言可變參數(shù)列表內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C++ String部分成員模擬實(shí)現(xiàn)流程詳解
我們先不直接實(shí)現(xiàn)完整版的string,先實(shí)現(xiàn)簡易版的string部分成員來基本了解下它的框架,以及以后來學(xué)習(xí)深淺拷貝的問題。這樣有循序漸進(jìn)的過程嘛2022-08-08C語言中g(shù)etch()函數(shù)詳解及簡單實(shí)例
這篇文章主要介紹了C語言中g(shù)etch()函數(shù)詳解及簡單實(shí)例的相關(guān)資料,需要的朋友可以參考下2017-03-03C語言創(chuàng)建動態(tài)dll和調(diào)用dll(visual studio 2013環(huán)境下)
本篇文章主要介紹了C語言創(chuàng)建動態(tài)dll和調(diào)用dll(visual studio 2013環(huán)境下),非常具有實(shí)用價(jià)值,需要的朋友可以參考下2017-11-11一起來了解一下C++的結(jié)構(gòu)體?struct
這篇文章主要為大家詳細(xì)介紹了C++的結(jié)構(gòu)體struct,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來幫助2022-02-02C++通過共享內(nèi)存ShellCode實(shí)現(xiàn)跨進(jìn)程傳輸
在計(jì)算機(jī)安全領(lǐng)域,ShellCode是一段用于利用系統(tǒng)漏洞或執(zhí)行特定任務(wù)的機(jī)器碼,本文主要為大家介紹了C++如何通過共享內(nèi)存ShellCode實(shí)現(xiàn)跨進(jìn)程傳輸,需要的可以參考下2023-12-12詳解如何實(shí)現(xiàn)C++虛函數(shù)調(diào)用匯編代碼
多態(tài)是C++中最重要的特性之一,對虛函數(shù)的調(diào)用在C++代碼中是隨處可見的,本篇文章我們詳細(xì)探討一下,感興趣的朋友快來看看吧2021-11-11Linux?C/C++實(shí)現(xiàn)網(wǎng)絡(luò)流量分析工具
網(wǎng)絡(luò)流量分析的原理基于對數(shù)據(jù)包的捕獲、解析和統(tǒng)計(jì)分析,通過對網(wǎng)絡(luò)流量的細(xì)致觀察和分析,幫助管理員了解和優(yōu)化網(wǎng)絡(luò)的性能,本文將通過C++實(shí)現(xiàn)網(wǎng)絡(luò)流量分析工具,有需要的可以參考下2023-10-10Qt實(shí)現(xiàn)指針式時(shí)鐘 Qt實(shí)現(xiàn)動態(tài)時(shí)鐘
這篇文章主要為大家詳細(xì)介紹了Qt實(shí)現(xiàn)指針式時(shí)鐘,Qt實(shí)現(xiàn)動態(tài)時(shí)鐘,兩者相互切換,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-07-07