C語言深入講解函數(shù)參數(shù)的使用
一、函數(shù)參數(shù)
- 函數(shù)參數(shù)在本質(zhì)上與局部變量相同在棧上分配空間
- 函數(shù)參數(shù)的初始值是函數(shù)調(diào)用時(shí)的實(shí)參值
- 函數(shù)參數(shù)的求值順序依賴于編譯器的實(shí)現(xiàn)

下面看一個(gè)函數(shù)參數(shù)的求值順序的示例:
#include <stdio.h>
int func(int i, int j)
{
printf("i = %d, j = %d\n", i, j);
return 0;
}
int main()
{
int k = 1;
func(k++, k++);
printf("%d\n", k);
return 0;
}
輸出結(jié)果如下:

這個(gè)示例說明函數(shù)參數(shù)的求值順序依賴于編譯器的實(shí)現(xiàn)。
二、程序的順序點(diǎn)
- 程序中存在一定的順序點(diǎn)
- 順序點(diǎn)指的是執(zhí)行過程中修改變量值的最晚時(shí)刻
- 在程序到達(dá)順序點(diǎn)的時(shí)候,之前所做的—切操作必須完成
- 每個(gè)完整表達(dá)式結(jié)束時(shí),即分號處
- &&,||,?:,以及逗號表達(dá)式的每個(gè)參數(shù)計(jì)算之后
- 函數(shù)調(diào)用時(shí)所有實(shí)參求值完成后(進(jìn)入函數(shù)體之前)
下面看一個(gè)程序中的順序點(diǎn)示例:
#include <stdio.h>
int main()
{
int k = 2;
int a = 1;
k = k++ + k++;
printf("k = %d\n", k);
if( a-- && a )
{
printf("a = %d\n", a);
}
return 0;
}
輸出結(jié)果如下:

a-- && a ,對于 && 運(yùn)算符,每個(gè)操作數(shù)都是一個(gè)順序點(diǎn)。當(dāng)程序從左往后執(zhí)行時(shí),a-- 對內(nèi)存的修改必須立即完成,所以 a 就變成了 0。
為什么會輸出 6 呢?下面在 VS2012 里面運(yùn)行代碼,進(jìn)行反匯編操作:

這段匯編代碼簡單的來說,就是先進(jìn)行 + 操作,k = 2 + 2 = 4,然后進(jìn)行兩次 ++ 操作,所以最終結(jié)果就是 6。
三、小結(jié)-上
- 函數(shù)的參數(shù)在棧上分配空間
- 函數(shù)的實(shí)參并沒有固定的計(jì)算次序
- 順序點(diǎn)是 C 語言中變量修改的最晚時(shí)機(jī)
四、調(diào)用約定
函數(shù)參數(shù)的計(jì)算次序是依賴編譯器實(shí)現(xiàn)的,那么函數(shù)參數(shù)的入棧次序是如何確定的呢?
當(dāng)函數(shù)調(diào)用發(fā)生時(shí)
- 參數(shù)會傳遞給被調(diào)用的函數(shù)
- 而返回值會被返回給函數(shù)調(diào)用者
調(diào)用約定描述參數(shù)如何傳遞到棧中以及棧的維護(hù)方式
- 參數(shù)傳遞順序
- 調(diào)用棧清理
調(diào)用約定是預(yù)定義的可理解為調(diào)用協(xié)議
調(diào)用約定通常用于庫調(diào)用和庫開發(fā)的時(shí)候
- 從右到左依次入棧:_stdcall,_cdecl,_thiscall
- 從左到右依次入棧:_pascal,_fastcall

五、可變參數(shù)
計(jì)算平均值時(shí),我們一般可以編寫成這樣:
#include <stdio.h>
float average(int array[], int size)
{
int i = 0;
float avr = 0;
for(i=0; i<size; i++)
{
avr += array[i];
}
return avr / size;
}
int main()
{
int array[] = {1, 2, 3, 4, 5};
printf("%f\n", average(array, 5));
return 0;
}
輸出結(jié)果如下:

C語言中可以定義參數(shù)可變的函數(shù)
參數(shù)可變函數(shù)的實(shí)現(xiàn)依賴于 stdarg.h 頭文件
- va_list -- 參數(shù)集合
- va_arg -- 取具體參數(shù)值
- va_start -- 標(biāo)識參數(shù)訪問的開始
- va_end -- 標(biāo)識參數(shù)訪問的結(jié)束
下面看一個(gè)求可變參數(shù)平均值的代碼:
#include <stdio.h>
#include <stdarg.h>
float average(int n, ...)
{
va_list args;
int i = 0;
float sum = 0;
va_start(args, n);
for(i=0; i<n; i++)
{
sum += va_arg(args, int);
}
va_end(args);
return sum / n;
}
int main()
{
printf("%f\n", average(5, 1, 2, 3, 4, 5));
printf("%f\n", average(4, 1, 2, 3, 4));
return 0;
}
輸出結(jié)果如下:

六、可變參數(shù)的限制
- 可變參數(shù)必須從頭到尾按照順序逐個(gè)訪問
- 參數(shù)列表中至少要存在一個(gè)確定的命名參數(shù)
- 可變參數(shù)函數(shù)無法確定實(shí)際存在的參數(shù)的數(shù)量
- 可變參數(shù)函數(shù)無法確定參數(shù)的實(shí)際類型
注意:va_arg 中如果指定了錯(cuò)誤的類型,那么結(jié)果是不可預(yù)測的。
七、小結(jié)-下
- 調(diào)用約定指定了函數(shù)參數(shù)的入棧順序以及棧的清理方式
- 可變參數(shù)是 C 語言提供的一種函數(shù)設(shè)計(jì)技巧
- 可變參數(shù)的函數(shù)提供了一種更方便的函數(shù)調(diào)用方式
- 可變參數(shù)必須順序的訪問,無法直接訪問中間的參數(shù)值
到此這篇關(guān)于C語言深入講解函數(shù)參數(shù)的使用的文章就介紹到這了,更多相關(guān)C語言 函數(shù)參數(shù)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C語言驅(qū)動開發(fā)之通過ReadFile與內(nèi)核層通信
驅(qū)動與應(yīng)用程序的通信是非常有必要的,內(nèi)核中執(zhí)行代碼后需要將其動態(tài)顯示給應(yīng)用層。為了實(shí)現(xiàn)內(nèi)核與應(yīng)用層數(shù)據(jù)交互則必須有通信的方法,微軟為我們提供了三種通信方式,本文先來介紹通過ReadFile系列函數(shù)實(shí)現(xiàn)的通信模式2022-09-09
C++基于socket編程實(shí)現(xiàn)聊天室功能
這篇文章主要介紹了C++基于socket編程實(shí)現(xiàn)聊天室功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-07-07
C語言線性表的順序表示與實(shí)現(xiàn)實(shí)例詳解
這篇文章主要介紹了C語言線性表的順序表示與實(shí)現(xiàn)實(shí)例詳解,對于學(xué)習(xí)數(shù)據(jù)結(jié)構(gòu)與算法的朋友很有參考借鑒價(jià)值,需要的朋友可以參考下2014-07-07
C++實(shí)現(xiàn)商店倉庫管理系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了C++實(shí)現(xiàn)商店倉庫管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-03-03

