C語(yǔ)言:在函數(shù)參數(shù)中傳遞指針的指針簡(jiǎn)析
這是道哥的第013篇原創(chuàng)
前言
今天同事問(wèn)了一個(gè)問(wèn)題:在函數(shù)參數(shù)中傳遞指針的指針,很常用的一個(gè)場(chǎng)景,重新梳理一下記錄于此,以后如果有類(lèi)似的問(wèn)題直接發(fā)這篇小總結(jié)就可以了。
代碼:版本1
void do_malloc(char *p, int size){ p = (char *)malloc(size + 1); memset(p, 0, size + 1);}
int main(int argc, char *argv[]){ char *pData = 0; do_malloc(pData, 128); sprintf(pData, "%s", "abc"); printf(pData); return 0;}
代碼本意是:do_work()函數(shù)向系統(tǒng)堆空間申請(qǐng)size個(gè)字節(jié)的空間,然后返回給main函數(shù)中的pData指針。但是,執(zhí)行的時(shí)候報(bào)錯(cuò):Segmentation fault (core dumped)。
分析原因
我們可以把char*類(lèi)型的指針看成一個(gè)遙控器,如果給這個(gè)指針賦值,就相當(dāng)于把這個(gè)遙控器與一個(gè)設(shè)備進(jìn)行綁定,可以通過(guò)遙控器來(lái)控制這個(gè)設(shè)備。
執(zhí)行char *pData = 0;
pData內(nèi)容為空,相當(dāng)于這個(gè)遙控器沒(méi)有與任何設(shè)備綁定,如下圖:
執(zhí)行do_work(pData, 128);
這里傳遞的參數(shù)是pData本身,所以進(jìn)入void do_work(char *p, int size)函數(shù)之后,實(shí)參pData的內(nèi)容就賦值給形參p,所以指針p的內(nèi)容也為空,也就是說(shuō):p這個(gè)遙控器也沒(méi)有與任何設(shè)備綁定,如下圖:
執(zhí)行p = (char *)malloc(size + 1);
這句話的作用是把申請(qǐng)到的堆空間的首地址,賦值給p。就是說(shuō):現(xiàn)在p指向了內(nèi)存中的一塊空間,就相當(dāng)于一個(gè)p這個(gè)遙控器與一個(gè)設(shè)備進(jìn)行綁定了,可以控制這個(gè)設(shè)備了,如下圖:
到這里就已經(jīng)看到程序崩潰的原因了:雖然給指針p賦值了,但是實(shí)參pData中的內(nèi)容一直為空,因此從do_malloc函數(shù)返回之后,pData仍然是一個(gè)空指針,所以就崩潰了。當(dāng)然,p指向的堆空間也就泄露了。
代碼:版本2
代碼的本意是在do_malloc函數(shù)中申請(qǐng)堆空間,然后把這塊空間的首地址賦值給pData。在do_malloc函數(shù)中,調(diào)用系統(tǒng)函數(shù)malloc成功之后返回所分配空間的首地址,關(guān)鍵是要把這個(gè)首地址送給pData指針,也就是說(shuō)要讓pData指針變量中的值等于這個(gè)堆空間的首地址。
那應(yīng)該如何通過(guò)中間的一個(gè)函數(shù)來(lái)完成這個(gè)功能呢,如下代碼:
void do_malloc(char **p, int size){ *p = (char *)malloc(size + 1); memset(*p, 0, size + 1);}
int main(int argc, char *argv[]){ char *pData = 0; do_malloc(&pData, 128); sprintf(pData, "%s", "abc"); printf(pData); return 0;}
執(zhí)行char *pData = 0;
這一句沒(méi)有變化。
執(zhí)行do_malloc(&pData, 128);
把pData指針的地址作為實(shí)參進(jìn)行傳遞,因?yàn)閜Data本身就是一個(gè)指針,加上取地址符&,就是指針的指針(二級(jí)指針),因此do_malloc函數(shù)的第一個(gè)參數(shù)就要定義成char**類(lèi)型,此時(shí)示意如圖:
p此時(shí)是一個(gè)二級(jí)指針,參數(shù)賦值之后,p里面的內(nèi)容就變成了pData這個(gè)指針變量的地址,也就是說(shuō)p指向了pData這個(gè)變量。
執(zhí)行*p = (char *)malloc(size + 1);
這句話首先搞明白*p是啥意思,剛才說(shuō)了,p是一個(gè)指針,它指向了pData這個(gè)變量。那么在p前面加上取值操作符*,就相當(dāng)于是取出指針p中的值,它里面的值就是pData!因此,malloc函數(shù)返回的堆空間首地址,就相當(dāng)于是賦值給了pData,如下圖:
此時(shí),pData這個(gè)遙控器就與分配的這塊堆空間綁定在一起,隨后再操作pData就沒(méi)有問(wèn)題了。

發(fā)表評(píng)論
請(qǐng)輸入評(píng)論內(nèi)容...
請(qǐng)輸入評(píng)論/評(píng)論長(zhǎng)度6~500個(gè)字
圖片新聞
-
馬云重返一線督戰(zhàn),阿里重啟創(chuàng)始人模式
-
機(jī)器人奧運(yùn)會(huì)戰(zhàn)報(bào):宇樹(shù)機(jī)器人摘下首金,天工Ultra搶走首位“百米飛人”
-
存儲(chǔ)圈掐架!江波龍起訴佰維,索賠121萬(wàn)
-
長(zhǎng)安汽車(chē)母公司突然更名:從“中國(guó)長(zhǎng)安”到“辰致科技”
-
豆包前負(fù)責(zé)人喬木出軌BP后續(xù):均被辭退
-
字節(jié)AI Lab負(fù)責(zé)人李航卸任后返聘,Seed進(jìn)入調(diào)整期
-
員工持股爆雷?廣汽埃安緊急回應(yīng)
-
中國(guó)“智造”背后的「關(guān)鍵力量」
最新活動(dòng)更多
-
10月23日火熱報(bào)名中>> 2025是德科技創(chuàng)新技術(shù)峰會(huì)
-
10月23日立即報(bào)名>> Works With 開(kāi)發(fā)者大會(huì)深圳站
-
10月24日立即參評(píng)>> 【評(píng)選】維科杯·OFweek 2025(第十屆)物聯(lián)網(wǎng)行業(yè)年度評(píng)選
-
即日-11.25立即下載>>> 費(fèi)斯托白皮書(shū)《柔性:汽車(chē)生產(chǎn)未來(lái)的關(guān)鍵》
-
11月27日立即報(bào)名>> 【工程師系列】汽車(chē)電子技術(shù)在線大會(huì)
-
12月18日立即報(bào)名>> 【線下會(huì)議】OFweek 2025(第十屆)物聯(lián)網(wǎng)產(chǎn)業(yè)大會(huì)
推薦專(zhuān)題
- 1 特斯拉工人被故障機(jī)器人打成重傷,索賠3.6億
- 2 【行業(yè)深度研究】退居幕后四年后,張一鳴終于把算法公司變成AI公司?
- 3 AI 時(shí)代,阿里云想當(dāng)“安卓” ,那誰(shuí)是“蘋(píng)果”?
- 4 華為公布昇騰芯片三年計(jì)劃,自研HBM曝光
- 5 硬剛英偉達(dá)!華為發(fā)布全球最強(qiáng)算力超節(jié)點(diǎn)和集群
- 6 機(jī)器人9月大事件|3家國(guó)產(chǎn)機(jī)器人沖刺IPO,行業(yè)交付與融資再創(chuàng)新高!
- 7 谷歌“香蕉”爆火啟示:國(guó)產(chǎn)垂類(lèi)AI的危機(jī)還是轉(zhuǎn)機(jī)?
- 8 00后華裔女生靠?jī)刹緼I電影狂賺7.8億人民幣,AI正式進(jìn)軍好萊塢
- 9 美光:AI Capex瘋投不止,終于要拉起存儲(chǔ)超級(jí)周期了?
- 10 華為已殺入!AI領(lǐng)域最熱黃金賽道,大廠的數(shù)字人美女讓我一夜沒(méi)睡著覺(jué)