C語(yǔ)言 用指針作為函數(shù)返回值詳解
C語(yǔ)言允許函數(shù)的返回值是一個(gè)指針(地址),我們將這樣的函數(shù)稱(chēng)為指針函數(shù)。下面的例子定義了一個(gè)函數(shù) strlong(),用來(lái)返回兩個(gè)字符串中較長(zhǎng)的一個(gè):
#include <stdio.h> #include <string.h> char *strlong(char *str1, char *str2){ if(strlen(str1) >= strlen(str2)){ return str1; }else{ return str2; } } int main(){ char str1[30], str2[30], *str; gets(str1); gets(str2); str = strlong(str1, str2); printf("Longer string: %s\n", str); return 0; }
運(yùn)行結(jié)果:
C Language↙
c.biancheng.net↙
Longer string: c.biancheng.net
用指針作為函數(shù)返回值時(shí)需要注意的一點(diǎn)是,函數(shù)運(yùn)行結(jié)束后會(huì)銷(xiāo)毀在它內(nèi)部定義的所有局部數(shù)據(jù),包括局部變量、局部數(shù)組和形式參數(shù),函數(shù)返回的指針請(qǐng)盡量不要指向這些數(shù)據(jù),C語(yǔ)言沒(méi)有任何機(jī)制來(lái)保證這些數(shù)據(jù)會(huì)一直有效,它們?cè)诤罄m(xù)使用過(guò)程中可能會(huì)引發(fā)運(yùn)行時(shí)錯(cuò)誤。請(qǐng)看下面的例子:
#include <stdio.h> int *func(){ int n = 100; return &n; } int main(){ int *p = func(), n; n = *p; printf("value = %d\n", n); return 0; }
運(yùn)行結(jié)果:
value = 100
n 是 func() 內(nèi)部的局部變量,func() 返回了指向 n 的指針,根據(jù)上面的觀點(diǎn),func() 運(yùn)行結(jié)束后 n 將被銷(xiāo)毀,使用 *p 應(yīng)該獲取不到 n 的值。但是從運(yùn)行結(jié)果來(lái)看,我們的推理好像是錯(cuò)誤的,func() 運(yùn)行結(jié)束后 *p 依然可以獲取局部變量 n 的值,這個(gè)上面的觀點(diǎn)不是相悖嗎?
為了進(jìn)一步看清問(wèn)題的本質(zhì),不妨將上面的代碼稍作修改,在第9~10行之間增加一個(gè)函數(shù)調(diào)用,看看會(huì)有什么效果:
#include <stdio.h> int *func(){ int n = 100; return &n; } int main(){ int *p = func(), n; printf("c.biancheng.net\n"); n = *p; printf("value = %d\n", n); return 0; }
運(yùn)行結(jié)果:
c.biancheng.net
value = -2
可以看到,現(xiàn)在 p 指向的數(shù)據(jù)已經(jīng)不是原來(lái) n 的值了,它變成了一個(gè)毫無(wú)意義的甚至有些怪異的值。與前面的代碼相比,該段代碼僅僅是在 *p 之前增加了一個(gè)函數(shù)調(diào)用,這一細(xì)節(jié)的不同卻導(dǎo)致運(yùn)行結(jié)果有天壤之別,究竟是為什么呢?
前面我們說(shuō)函數(shù)運(yùn)行結(jié)束后會(huì)銷(xiāo)毀所有的局部數(shù)據(jù),這個(gè)觀點(diǎn)并沒(méi)錯(cuò),大部分C語(yǔ)言教材也都強(qiáng)調(diào)了這一點(diǎn)。但是,這里所謂的銷(xiāo)毀并不是將局部數(shù)據(jù)所占用的內(nèi)存全部抹掉,而是程序放棄對(duì)它的使用權(quán)限,棄之不理,后面的代碼可以隨意使用這塊內(nèi)存。對(duì)于上面的兩個(gè)例子,func() 運(yùn)行結(jié)束后 n 的內(nèi)存依然保持原樣,值還是 100,如果使用及時(shí)也能夠得到正確的數(shù)據(jù),如果有其它函數(shù)被調(diào)用就會(huì)覆蓋這塊內(nèi)存,得到的數(shù)據(jù)就失去了意義。
關(guān)于函數(shù)調(diào)用的原理以及函數(shù)如何占用內(nèi)存的更多細(xì)節(jié),我們將在《C語(yǔ)言和內(nèi)存》專(zhuān)題中深入探討,相信你必將有所頓悟,解開(kāi)心中的謎團(tuán)。
第一個(gè)例子在調(diào)用其他函數(shù)之前使用 *p 搶先獲得了 n 的值并將它保存起來(lái),第二個(gè)例子顯然沒(méi)有抓住機(jī)會(huì),有其他函數(shù)被調(diào)用后才使用 *p 獲取數(shù)據(jù),這個(gè)時(shí)候已經(jīng)晚了,內(nèi)存已經(jīng)被后來(lái)的函數(shù)覆蓋了,而覆蓋它的究竟是一份什么樣的數(shù)據(jù)我們無(wú)從推斷(一般是一個(gè)沒(méi)有意義甚至有些怪異的值)。
以上就是對(duì) C語(yǔ)言指針作為函數(shù)返回值的資料整理,后續(xù)繼續(xù)補(bǔ)充相關(guān)資料,謝謝大家對(duì)本站的支持!
相關(guān)文章
C++實(shí)現(xiàn)簡(jiǎn)易反彈小球游戲的示例代碼
我們利用printf 函數(shù)實(shí)現(xiàn)一個(gè)在屏幕上彈跳的小球。彈跳的小球游戲比較簡(jiǎn)單、容易入門(mén),也是反彈球消磚塊、接金幣、臺(tái)球等很多游戲的基礎(chǔ),感興趣的可以了解一下2022-10-10C語(yǔ)言超詳細(xì)講解猜數(shù)字游戲的實(shí)現(xiàn)
現(xiàn)在很多游戲都有抽獎(jiǎng)抽卡的功能,其實(shí)這個(gè)就類(lèi)似于猜數(shù)字,生成一個(gè)隨機(jī)數(shù),然后你去猜,猜對(duì)了就得獎(jiǎng)。猜到一定次數(shù)就會(huì)保底。要實(shí)現(xiàn)猜數(shù)字的小游戲,首先是要讓程序生成隨機(jī)數(shù),這就要用到rand、srand和time這三個(gè)函數(shù),其次要了解時(shí)間戳2022-07-07C++二叉樹(shù)的前序中序后序非遞歸實(shí)現(xiàn)方法詳細(xì)講解
前序遍歷的順序是根、左、右。任何一顆樹(shù)都可以認(rèn)為分為左路節(jié)點(diǎn),左路節(jié)點(diǎn)的右子樹(shù)。先訪(fǎng)問(wèn)左路節(jié)點(diǎn),再來(lái)訪(fǎng)問(wèn)左路節(jié)點(diǎn)的右子樹(shù)。把訪(fǎng)問(wèn)左路節(jié)點(diǎn)的右子樹(shù)看成一個(gè)子問(wèn)題,就可以完整遞歸訪(fǎng)問(wèn)了2023-03-03C++實(shí)現(xiàn)拼圖游戲代碼(graphics圖形庫(kù))
這篇文章主要為大家詳細(xì)介紹了C++實(shí)現(xiàn)拼圖游戲代碼,帶有g(shù)raphics圖形庫(kù),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-05-05C++實(shí)現(xiàn)紅黑樹(shù)應(yīng)用實(shí)例代碼
紅黑樹(shù)它一種特殊的二叉查找樹(shù),這意味著它滿(mǎn)足二叉查找樹(shù)的特征,但是也有許多自己的特性,這篇文章主要給大家介紹了關(guān)于C++實(shí)現(xiàn)紅黑樹(shù)的相關(guān)資料,需要的朋友可以參考下2021-11-11