欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C語言可變參數(shù)列表的用法與深度剖析

 更新時(shí)間:2022年02月07日 15:37:43   作者:^jhao^  
這篇文章主要給大家介紹了關(guān)于C語言可變參數(shù)列表的相關(guān)資料,文中通過實(shí)例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下

前言

可變參數(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)文章

最新評論