C語(yǔ)言參數(shù)傳遞
前言
最近復(fù)習(xí)Block相關(guān)知識(shí)晓淀,其中有個(gè)問(wèn)題:block中為什么不能改變(這里值重新復(fù)制)被截獲的自動(dòng)變量,這個(gè)問(wèn)題涉及到C語(yǔ)言參數(shù)傳遞,所以回頭看了一下典尾,并做了一些整理墨技。
下面是C語(yǔ)言的幾種參數(shù)傳遞方式:值傳遞惩阶、引用傳遞、指針傳遞扣汪、指針引用傳遞断楷。
PS:為方便表述,以下內(nèi)容的圖片對(duì)內(nèi)存的描述作了簡(jiǎn)化崭别,如int類型是4個(gè)字節(jié)冬筒,指針類型是8個(gè)字節(jié),這里統(tǒng)一用相同大小的一個(gè)格子表示茅主。
1. 值傳遞
代碼:
void func(int p)
{
printf("\n-----------func start-----------");
printf("\n&p : %x", &p);
printf("\np : %x", p);
p = 0xff;
printf("\n-----------func end-----------\n");
}
void test()
{
int a = 0x10;
printf("\n&a : %x", &a);
printf("\na : %x", a);
func(a);
printf("\na : %x", a);
}
輸出:
&a : efbfde3c
a : 10
-----------func start-----------
&p : efbfde1c
p : 10
-----------func end-----------
a : 10
通過(guò)上面的代碼及輸出可知舞痰,變量a的地址為efbfde3c
,值為10
诀姚;通過(guò)調(diào)用func(a)
响牛,通過(guò)輸出可知:變量p的地址為efbfde1c
,值為10
学搜。也就是說(shuō)參數(shù)傳遞只把變量a的值傳遞給了變量p娃善,此時(shí)變量a與變量p是值相同的兩個(gè)不同變量,在內(nèi)存不同的兩個(gè)地址中瑞佩,所以對(duì)變量p賦值只是修改了變量p的值聚磺,并不能修改變量a的值,如下圖所示炬丸√鼻蓿總結(jié)一句話就是:值傳遞不傳遞變量本身,只傳遞變量的值稠炬。
2.引用傳遞
代碼:
void func(int &p)
{
printf("\n-----------func start-----------");
printf("\n&p : %x", &p);
printf("\np : %x", p);
p = 0xff;
printf("\n-----------func end-----------\n");
}
void test()
{
int a = 0x10;
printf("&a : %x", &a);
printf("\na : %x\n", a);
func(a);
printf("\na : %x", a);
}
輸出:
&a : efbfde2c
a : 10
-----------func start-----------
&p : efbfde2c
p : 10
-----------func end-----------
a : ff
與值傳遞不同焕阿,func的參數(shù)是int &p
,表示引用傳遞首启。如輸出所示暮屡,通過(guò)引用傳遞,變量a的地址與變量p是相同的毅桃,也就是說(shuō)引用傳遞把a(bǔ)的地址傳給了變量p褒纲,所以對(duì)變量p進(jìn)行賦值之后准夷,變量a的值也改變了,因?yàn)樽兞縜與變量p實(shí)際是同一個(gè)變量莺掠。如下圖所示:
3. 指針傳遞
代碼:
void func(int *p)
{
printf("\n-----------func start-----------");
printf("\n&p : %x", &p);
printf("\n*p : %x", *p);
printf("\np : %x", p);
*p = 0xff;
printf("\n-----------func end-----------\n");
}
void test()
{
int a = 0x10;
printf("&a : %x", &a);
printf("\na : %x\n", a);
func(&a);
printf("\na : %x", a);
}
輸出:
&a : efbfde2c
a : 10
-----------func start-----------
&p : efbfde08
*p : 10
p : efbfde2c
-----------func end-----------
a : ff
指針傳遞衫嵌,顧名思義,int *p
表明p是一個(gè)指針變量彻秆,調(diào)用func(&a)
是把變量a的內(nèi)存地址傳遞過(guò)去賦值給指針變量p楔绞,此時(shí)指針變量p的值是變量a的內(nèi)存地址,故可看到輸出p為efbfde2c
(變量a的內(nèi)存地址)唇兑。*p
是對(duì)指針p的值(即變量a的內(nèi)存地址)進(jìn)行取值酒朵,相當(dāng)于*&a
,所以對(duì)*p
進(jìn)行賦值也就改變了變量a的值幔亥。如下圖所示:
4. 指針引用傳遞
代碼:
void func(int *&p)
{
printf("\n-----------func start-----------");
printf("\n&p : %x", &p);
printf("\n*p : %x", *p);
printf("\np : %x", p);
*p = 0xff;
printf("\n-----------func end-----------\n");
}
void test()
{
int a = 0x10;
printf("&a : %x", &a);
printf("\na : %x\n", a);
int *b = &a;
printf("\n&b : %x", &b);
printf("\n*b : %x", *b);
printf("\nb : %x", b);
func(b);
printf("\na : %x", a);
}
輸出:
&a : efbfde2c
a : 10
&b : efbfde20
*b : 10
b : efbfde2c
-----------func start-----------
&p : efbfde20
*p : 10
p : efbfde2c
-----------func end-----------
a : ff
其實(shí)理解了前面的耻讽,這里也就明了了察纯。首先把變量a
的地址即&a
賦值給指針變量b帕棉,指針變量b的值就是efbfde2c
(后面的的輸出證明了這一點(diǎn))。然后調(diào)用
func(b)
饼记,這里跟前面的引用傳遞一樣道理香伴,指針變量p實(shí)質(zhì)是是指針變量b,而指針變量b的值又是變量a的內(nèi)存地址具则,所以*p
等同于*b
即纲,對(duì)其進(jìn)行賦值相當(dāng)于對(duì)變量a進(jìn)行賦值(a = 0xff
)〔├撸可見(jiàn)下圖: