如何將一維數(shù)組作為函數(shù)形參來(lái)使用
作者:王姍姍,華清遠(yuǎn)見嵌入式學(xué)院講師。
請(qǐng)看下面的程序:
#include <stdio.h>
void func1(int arr[])
{
printf("%dn",sizeof(arr));
arr=arr+1;
printf("%dn",*arr);
}
void func2(int *arr)
{
printf("%dn",sizeof(arr));
arr=arr+1;
printf("%dn",*arr);
}
int main()
{
int a[10]={1,2,3,4,5};
int *b=a;
printf("%dn",sizeof(a));
func1(a);
func2(a);
printf("%dn",sizeof(b));
func1(b);
func2(b);
//a=a+1;
printf("%dn",*a);
b=b+1;
printf("%dn",*a);
return 0;
}
為了說(shuō)明問(wèn)題,我寫了一個(gè)簡(jiǎn)單的測(cè)試程序,在主函數(shù)中定義了一個(gè)a數(shù)組,并定義了一個(gè)b指針指向該數(shù)組的首地址。顯然a=a+1是不能通過(guò)編譯的,因?yàn)樽鳛閿?shù)組而言,a是地址常量,一旦定義是不允許指向其他地址的。
那么作為b,其僅僅是一個(gè)指針變量,里面存放的是a數(shù)組的首地址,故可以執(zhí)行b=b+1,這樣使得b指向a+1,也就是a數(shù)組中的a[1]。
下面問(wèn)題就來(lái)了,我們?cè)撊绾螌⒁粋€(gè)數(shù)組作為形參傳遞給函數(shù),也就是說(shuō)若有一個(gè)實(shí)參數(shù)組,想在函數(shù)中改變此數(shù)組中的元素的值,我們?cè)撊绾翁幚恚?/SPAN>
數(shù)組作為形參,和接受它的實(shí)參有以下的關(guān)系:
1. 形參和實(shí)參都是數(shù)組名。
2. 實(shí)參用數(shù)組,形參用指針變量.
3. 實(shí)參、形參都用指針變量。
4. 實(shí)參為指針變量,形參為數(shù)組名。
在上述的例子中我們來(lái)進(jìn)一步的講解。
首先來(lái)看結(jié)果:
40//sizeof(a)
4//func1(a)
2//arr=arr+1
4//func2(a)
2//arr=arr+1
4//sizeof(b)
4//func1(b)
2// arr=arr+1
4//func2(b)
2// arr=arr+1
1// a
1//b=b+1;
在C語(yǔ)言中,數(shù)組名是當(dāng)作指針來(lái)處理的。更確切的說(shuō),數(shù)組名就是指向數(shù)組首元素地址的指針,數(shù)組索引就是距數(shù)組首元素地址的偏移量。理解這一點(diǎn)很重要,很多數(shù)組應(yīng)用的問(wèn)題就是有此而起的。這也就是為什么C語(yǔ)言中的數(shù)組是從0開始計(jì)數(shù),因?yàn)檫@樣它的索引就比較好對(duì)應(yīng)到偏移量上。在C語(yǔ)言中,編譯過(guò)程中遇到有數(shù)組名的表達(dá)式,都會(huì)把數(shù)組名替換成指針來(lái)處理;編譯器甚至無(wú)法區(qū)分a[4]和4[a]的區(qū)別。
我們?cè)陬}中定義了
int a[10];
int *b;
這兩者并不等價(jià),第一句話聲明了數(shù)組a,并定義了這個(gè)數(shù)組,它有10個(gè)int型元素,sizeof(a)將得到整個(gè)數(shù)組所占的內(nèi)存大小,是40;第二句話只是聲明并定義了一個(gè)int型的指針,sizeof(b)將得到這個(gè)指針?biāo)嫉膬?nèi)存大小,是4。所以說(shuō),雖然數(shù)組名在表達(dá)式中一般會(huì)當(dāng)作指針來(lái)處理,但是數(shù)組名和指針還是有差距的,最起碼有a==&a[0]但是sizeof(a)!=sizeof(a[0])。
并且在ANSI C標(biāo)準(zhǔn)中,也明文規(guī)定:在函數(shù)參數(shù)的聲明中,數(shù)組名可以當(dāng)作指向該數(shù)組第一個(gè)元素的指針。所以,下面的幾種書寫形式是等效的:
void func1(int arr[]){}
void func2(int *arr){}
對(duì)于第一種還可以寫成void func1(int arr[100]){},在形參中其實(shí)不需要指定數(shù)組的大小,因?yàn)槠鋵?shí)在這幾個(gè)函數(shù)在函數(shù)調(diào)用時(shí)傳入的只是一個(gè)該數(shù)組的指針,想要確定幾行幾列的話還需要另外定義參數(shù)進(jìn)行傳入.如果在使用該指針的過(guò)程中不清楚原數(shù)組的范圍,指針很容易就越界,內(nèi)存也就溢出了。
當(dāng)用數(shù)組名作函數(shù)參數(shù)時(shí),形參數(shù)組就可以從實(shí)參數(shù)組那里得到起始地址后,形參數(shù)組就和實(shí)參數(shù)組共占同一段內(nèi)存單元,從而在調(diào)用函數(shù)期間,如果改變了形參數(shù)組的值,也就該變了實(shí)參數(shù)組的值。
最后要補(bǔ)充一點(diǎn),也就是上次我們學(xué)員問(wèn)我的一個(gè)問(wèn)題,作為形參int arr[]和int *arr有什么區(qū)別。在本質(zhì)上,我們看通過(guò)程序,貌似真的沒有區(qū)別,雖然arr[]是作為數(shù)組傳遞進(jìn)去,但是我們也可以修改arr的指向,而作為實(shí)參的數(shù)組的a卻不能使其加一。我們都知道函數(shù)之間的參數(shù)傳遞有:復(fù)制傳遞方式、地址傳遞方式、全局變量。
那么姑且就將void func1(int arr[])最為函數(shù)按復(fù)制傳遞的方式,而void func2(int *arr)作為其地址傳遞的方式。而復(fù)制傳遞方式與數(shù)組的指針傳遞方式是完全相同,只是形參的說(shuō)明形式不同而已。調(diào)用函數(shù)將實(shí)參數(shù)組傳遞給被調(diào)用函數(shù)形參,形參接收是實(shí)參的地址。不論哪種方式,被調(diào)用函數(shù)里對(duì)形參數(shù)組的操作都會(huì)影響調(diào)用函數(shù)里的實(shí)參數(shù)組。
“本文由華清遠(yuǎn)見http://www.embedu.org/index.htm提供”
來(lái)源:華清遠(yuǎn)見0次