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

C語(yǔ)言可變參數(shù)與函數(shù)參數(shù)的內(nèi)存對(duì)齊詳解

 更新時(shí)間:2022年03月28日 10:32:35   作者:熠熠L  
這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言可變參數(shù)與函數(shù)參數(shù)的內(nèi)存對(duì)齊,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來(lái)幫助

什么是可變參數(shù)?

有時(shí),您可能會(huì)碰到這樣的情況,您希望函數(shù)帶有可變數(shù)量的參數(shù),而不是預(yù)定義數(shù)量的參數(shù)。

C 語(yǔ)言為這種情況提供了一個(gè)解決方案,它允許您定義一個(gè)函數(shù),能根據(jù)具體的需求接受可變數(shù)量的參數(shù)。

比如我們最常用的printf函數(shù),它的函數(shù)聲明是:int printf(const char *format, ...);該函數(shù)就是一個(gè)典型的應(yīng)用可變參數(shù)的實(shí)例,后面那三個(gè)...就是說(shuō)明該函數(shù)是可變參數(shù)函數(shù)。

使用可變參數(shù)

要使用可變函數(shù),得引用一個(gè)頭文件#include <stdarg.h>該文件提供了實(shí)現(xiàn)可變參數(shù)功能的函數(shù)和宏。
使用可變參數(shù)的步驟如下:

1.定義一個(gè)函數(shù),最后一個(gè)參數(shù)為省略號(hào)...,省略號(hào)前面可以設(shè)置自定義參數(shù)(至少得有一個(gè)固定參數(shù))。如
int getSum(int num, ...)//定義可變參數(shù)的函數(shù)

2.在函數(shù)中定義va_list類型的變量list,該類型在stdarg.h中已定義。

3.使用宏函數(shù)va_start來(lái)初始化變量list,該宏函數(shù)在stdarg.h中已定義。

4.使用宏函數(shù)va_arglist來(lái)訪問(wèn)參數(shù)列表中的每個(gè)項(xiàng)。

5.使用宏函數(shù)va_end來(lái)清理賦予list變量的內(nèi)存。

宏的聲明

/** \brief      初始化 ap 變量,它與 va_arg 和 va_end 宏是一起使用的。
 *				last_arg 是最后一個(gè)傳遞給函數(shù)的已知的固定參數(shù),即省略號(hào)之前的參數(shù)。
 *				這個(gè)宏必須在使用 va_arg 和 va_end 之前被調(diào)用。
 *
 * \param   	ap -- 這是一個(gè) va_list 類型的對(duì)象,
 * 						它用來(lái)存儲(chǔ)通過(guò) va_arg 獲取額外參數(shù)時(shí)所必需的信息。
 * \param   	last_arg -- 最后一個(gè)傳遞給函數(shù)的已知的固定參數(shù)(省略號(hào)前面的那個(gè)參數(shù))。
 * \return      無(wú)
 *
 */
void va_start(va_list ap, last_arg)

/** \brief      檢索函數(shù)參數(shù)列表中類型為 type 的下一個(gè)參數(shù)。它無(wú)法判斷檢索到的參數(shù)是否是傳給函數(shù)的最后一個(gè)參數(shù)。
 *
 * \param   	ap -- 這是一個(gè) va_list 類型的對(duì)象,存儲(chǔ)了有關(guān)額外參數(shù)和檢索狀態(tài)的信息。
 * 					該對(duì)象應(yīng)在第一次調(diào)用 va_arg 之前通過(guò)調(diào)用 va_start 進(jìn)行初始化。
 * \param   	type -- 這是一個(gè)類型名稱。該類型名稱是作為擴(kuò)展自該宏的表達(dá)式的類型來(lái)使用的。
 * \return      該宏返回下一個(gè)額外的參數(shù),是一個(gè)類型為 type 的表達(dá)式。
 *
 */
type va_arg(va_list ap, type)

/** \brief  	該宏允許使用了 va_start 宏的帶有可變參數(shù)的函數(shù)返回(釋放內(nèi)存)。如果在從函數(shù)返回之前沒(méi)有調(diào)用 va_end,則結(jié)果為未定義。
 *
 * \param   	ap -- 這是之前由同一函數(shù)中的 va_start 初始化的 va_list 對(duì)象。
 * \return      無(wú)
 *
 */
void va_end(va_list ap)

實(shí)例1一個(gè)可變參數(shù)的函數(shù),求和

#include <stdio.h>
#include <stdarg.h>//引用可變參數(shù)宏頭文件

int getSum(int num, ...)//定義可變參數(shù)的函數(shù)
{
    int sum = 0;
    va_list list;//創(chuàng)建va_list類型的變量
    va_start(list, num);//初始化可變參數(shù)list
    for(int i = 0; i < num; i++)
    {
        sum += va_arg(list, int);//訪問(wèn)參數(shù)列表中的每個(gè)項(xiàng)
    }
    va_end(list);//釋放內(nèi)存
    return sum;
}

int main()
{
    printf("%d\n", getSum(4, 4, 5, 6, 7));
}

實(shí)例2輸出字符串

#include <stdio.h>
#include <stdarg.h>//引用可變參數(shù)宏頭文件

void func(char *demo, ...)
{
    char *pstr = NULL;
    va_list list;
    va_start(list, demo);
    while(1)
    {
        pstr = va_arg(list, char *);
        if(*pstr == '$')//以 '$' 代表結(jié)束
            break;
        printf("%s\n", pstr);
    }
    va_end(list);
}

int main()
{
    func("demo", "ABC", "123", "Hello Wolrd!", '$');
}

這里特別注意一下,宏va_arg無(wú)法判斷檢索到的參數(shù)是否是傳給函數(shù)的最后一個(gè)參數(shù),所以我們需要告訴該參數(shù)是不是最后一個(gè)參數(shù),有2個(gè)方法,一是在使用一個(gè)函數(shù)參數(shù)來(lái)說(shuō)明可變參數(shù)的數(shù)量,一是定義一個(gè)結(jié)束標(biāo)志符。

可變參數(shù)的另外的一種使用方式

#include <stdio.h>

int getSum(int num, ...)
{
    int sum = 0;
    char *p = NULL;
    p = (char*)&num;
    p += 8;
    for(int i = 0; i < num; i++)
    {
        sum += *((int*)p);
        p += 8;
    }
    return sum;
}
int main()
{
    int a = 1;
    int b = 2;
    int c = 3;
    printf("sum = %d\n", getSum(3, a, b, c));
}
/*
輸出結(jié)果
sum = 6;
*/

為什么這樣也可以訪問(wèn)可變參數(shù)呢?為什么指針p要加8呢?
因?yàn)檫@與函數(shù)參數(shù)的入棧出棧及函數(shù)參數(shù)的內(nèi)存對(duì)齊有關(guān)。

函數(shù)參數(shù)的內(nèi)存對(duì)齊

首先我們來(lái)看函數(shù)void func(int a, int b, int c)各個(gè)參數(shù)在棧中的位置

c高地址
b
a低地址

函數(shù)參數(shù)的傳遞存儲(chǔ)在棧中,從右至左壓入棧中,壓棧過(guò)程為遞減;出棧過(guò)程為遞增。

所以我們只需要知道a的地址,在a的地址上加上偏移量就可以訪問(wèn)b或者c了。

那應(yīng)該加上多少偏移量呢?

#include <stdio.h>

void func(int a, int b, int c)
{
    printf("a = %p\n", &a);
    printf("b = %p\n", &b);
    printf("c = %p\n", &c);
}
int main()
{
	int a,b,c;
	func(a, b, c);
}
/*
輸出結(jié)果
a = 000000000061FDF0
b = 000000000061FDF8
c = 000000000061FE00
*/

通過(guò)上例,發(fā)現(xiàn)它們之間相差8,為什么是8呢?

因?yàn)槲沂窃赪indow64位上運(yùn)行的,故需要按照8字節(jié)對(duì)齊。

綜上,函數(shù)參數(shù)的傳遞存儲(chǔ)在棧中,從右至左壓入棧中,壓棧過(guò)程為遞減,出棧過(guò)程為遞增;并且需要進(jìn)行內(nèi)存對(duì)齊,Window64位為8字節(jié)對(duì)齊,32位為4字節(jié)對(duì)齊。

下面是我做的一些實(shí)驗(yàn),更改了函數(shù)參數(shù)類型。

短整型

#include <stdio.h>

void func(char a, short b, long long c)
{
    printf("a = %p\n", &a);
    printf("b = %p\n", &b);
    printf("c = %p\n", &c);
}

int main()
{
    char a = 1;
    short b = 2;
    long long c = 3;
    func(a, b, c);
}
/*
輸出結(jié)果
a = 000000000061FDF0
b = 000000000061FDF8
c = 000000000061FE00
*/

浮點(diǎn)型

#include <stdio.h>

void func(double a, double b, double c)
{
    printf("a = %p\n", &a);
    printf("b = %p\n", &b);
    printf("c = %p\n", &c);
}

int main()
{
    double a = 1;
    double b = 2;
    double c = 3;
    func(a, b, c);
}
/*
輸出結(jié)果
a = 000000000061FDF0
b = 000000000061FDF8
c = 000000000061FE00
*/

結(jié)構(gòu)體1

#include <stdio.h>
typedef struct
{
    char c[7];
}str_t;

void func(str_t a, str_t b, str_t c)
{
    printf("a = %p\n", &a);
    printf("b = %p\n", &b);
    printf("c = %p\n", &c);
}

int main()
{
    str_t a;
    str_t b;
    str_t c;
    func(a, b, c);
}
/*
輸出結(jié)果
a = 000000000061FDF0
b = 000000000061FDE0
c = 000000000061FDD0
*/

結(jié)構(gòu)體2

#include <stdio.h>
typedef struct
{
    char c[4];
}str_t;

void func(str_t a, str_t b, str_t c)
{
    printf("a = %p\n", &a);
    printf("b = %p\n", &b);
    printf("c = %p\n", &c);
}

int main()
{
    str_t a;
    str_t b;
    str_t c;
    func(a, b, c);
}
/*
輸出結(jié)果
a = 000000000061FDF0
b = 000000000061FDF8
c = 000000000061FE00
*/

結(jié)構(gòu)體1出問(wèn)題了,具體沒(méi)搞明白,歡迎大佬指導(dǎo)。

建議可變參數(shù)使用引用頭文件stdarg.h得方式較好。

總結(jié)

本篇文章就到這里了,希望能夠給你帶來(lái)幫助,也希望您能夠多多關(guān)注腳本之家的更多內(nèi)容! 

相關(guān)文章

  • 使用C語(yǔ)言實(shí)現(xiàn)本地socke通訊的方法

    使用C語(yǔ)言實(shí)現(xiàn)本地socke通訊的方法

    這篇文章主要介紹了?使用C語(yǔ)言實(shí)現(xiàn)本地socke通訊,代碼分為服務(wù)器代碼和客戶端代碼,代碼簡(jiǎn)單易懂,對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2021-12-12
  • Visual Studio Code運(yùn)行C++代碼時(shí)顯示CLOCKS_PER_SEC未定義的問(wèn)題及解決方法

    Visual Studio Code運(yùn)行C++代碼時(shí)顯示CLOCKS_PER_SEC未定義的問(wèn)題及解決方法

    這篇文章主要介紹了解決Visual Studio Code運(yùn)行C++代碼時(shí)顯示CLOCKS_PER_SEC未定義的問(wèn)題,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-04-04
  • C++中的各種容器的使用方法匯總

    C++中的各種容器的使用方法匯總

    這篇文章主要介紹了C++中的各種容器的使用方法,本文結(jié)合示例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2023-01-01
  • C++數(shù)據(jù)封裝以及定義結(jié)構(gòu)的詳細(xì)講解

    C++數(shù)據(jù)封裝以及定義結(jié)構(gòu)的詳細(xì)講解

    這篇文章主要詳細(xì)講解了C++數(shù)據(jù)封裝以及定義結(jié)構(gòu),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2023-05-05
  • C++基于easyx圖形庫(kù)實(shí)現(xiàn)推箱子游戲

    C++基于easyx圖形庫(kù)實(shí)現(xiàn)推箱子游戲

    這篇文章主要為大家詳細(xì)介紹了C++基于easyx圖形庫(kù)實(shí)現(xiàn)推箱子游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2020-06-06
  • C++ 數(shù)據(jù)結(jié)構(gòu)線性表-數(shù)組實(shí)現(xiàn)

    C++ 數(shù)據(jù)結(jié)構(gòu)線性表-數(shù)組實(shí)現(xiàn)

    這篇文章主要介紹了C++ 數(shù)據(jù)結(jié)構(gòu)線性表-數(shù)組實(shí)現(xiàn)的相關(guān)資料,需要的朋友可以參考下
    2017-06-06
  • C語(yǔ)言中數(shù)據(jù)如何存儲(chǔ)進(jìn)內(nèi)存揭秘

    C語(yǔ)言中數(shù)據(jù)如何存儲(chǔ)進(jìn)內(nèi)存揭秘

    使用編程語(yǔ)言進(jìn)行編程時(shí),需要用到各種變量來(lái)存儲(chǔ)各種信息。變量保留的是它所存儲(chǔ)的值的內(nèi)存位置。這意味著,當(dāng)您創(chuàng)建一個(gè)變量時(shí),就會(huì)在內(nèi)存中保留一些空間。您可能需要存儲(chǔ)各種數(shù)據(jù)類型的信息,操作系統(tǒng)會(huì)根據(jù)變量的數(shù)據(jù)類型,來(lái)分配內(nèi)存和決定在保留內(nèi)存中存儲(chǔ)什么
    2022-08-08
  • C語(yǔ)言詳細(xì)分析宏定義的使用

    C語(yǔ)言詳細(xì)分析宏定義的使用

    宏定義是用宏名來(lái)表示一個(gè)字符串,在宏展開(kāi)時(shí)又以該字符串取代宏名,這只是一種簡(jiǎn)單的替換。字符串中可以含任何字符,可以是常數(shù),也可以是表達(dá)式,預(yù)處理程序?qū)λ蛔魅魏螜z查,如有錯(cuò)誤,只能在編譯已被宏展開(kāi)后的源程序時(shí)發(fā)現(xiàn)
    2022-04-04
  • C++替換棧中和.data中的cookie實(shí)現(xiàn)步驟詳解

    C++替換棧中和.data中的cookie實(shí)現(xiàn)步驟詳解

    這篇文章主要介紹了C++替換棧中和.data中的cookie實(shí)現(xiàn)步驟,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)吧
    2022-10-10
  • Qt利用QState狀態(tài)機(jī)實(shí)現(xiàn)控件互斥操作詳解

    Qt利用QState狀態(tài)機(jī)實(shí)現(xiàn)控件互斥操作詳解

    這篇文章主要為大家詳細(xì)介紹了Qt如何利用QState狀態(tài)機(jī)實(shí)現(xiàn)控件互斥操作,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下
    2022-12-12

最新評(píng)論