C語言深入講解之從函數(shù)棧幀角度理解return關(guān)鍵字
初識函數(shù)棧幀

如上圖可見,函數(shù)在被調(diào)用的時候會現(xiàn)在棧上開辟一個空間,我們稱之為棧幀,之后函數(shù)內(nèi)部的變量在這塊區(qū)域進行空間開辟。
但是函數(shù)在調(diào)用的時候,怎么知道需要開辟多大空間呢???
void func()
{
int a, b;
double c, d, e;
}
按照示例代碼,會先對需要的內(nèi)存空間大小進行預(yù)估,然后進行空間開辟。
函數(shù)返回時,棧幀會被釋放,但是,雖然棧幀被釋放,里面的內(nèi)容是不會被清空的,下面通過以下的例子進行分析。
#include <stdio.h>
#include <windows>
char* show()
{
char str[] = "hello world!";
return str;
}
int main()
{
char* s = show();
printf("%s\n", s);
system("pause");
return 0;
}
運行會得到亂碼的結(jié)果

看到這里,有些小伙伴肯定會說,嗷,return語句是不可以返回指向棧內(nèi)存的指針的。
可是這又是為什么呢?
于是我按下F11進行調(diào)試。發(fā)現(xiàn)當代碼進行到printf語句行的時候,s指向的內(nèi)容依舊是hello world!。繼續(xù)F10,到14行的時候,printf函數(shù)被調(diào)用,s字符串居然又不存在了?。。?!
其實,printf也是個函數(shù),也會在調(diào)用函數(shù)的時候形成棧幀,會覆蓋曾經(jīng)show棧幀存在的位置,而show棧幀在被釋放之后是無效的。
return
接下來,讓我們把關(guān)注點放到return關(guān)鍵字,同樣的,從代碼出發(fā)~~

誒,那就奇怪了???上面不是說過函數(shù)棧幀會被釋放嗎,那x的值又是怎么被y拿到的呀???
函數(shù)的返回值其實是通過寄存器的方式返回給調(diào)用方的
同樣的,讓我們看看調(diào)試。

這是進入了GetData函數(shù)的匯編語言,eax其實就是寄存器,14行下一行的意思就是把x的內(nèi)容放入寄存器里。
然后我們繼續(xù)F11

回到了main函數(shù),發(fā)現(xiàn)eax會把值再次放入y中。
看到這里是不是恍然大悟了呢?。?!
來看另外一種情況,如果返回的值不被接收呢???

如果返回的值不被接收,GetData后續(xù)沒有處理eax。
個人總結(jié)環(huán)節(jié)
最后,來把知識點系統(tǒng)回顧一遍!
return返回值本質(zhì)上是通過寄存器返回的,如果返回的是一個值,在有變量接收該返回類型的情況下,可以打印該數(shù)據(jù)。如果返回的是一個指針,雖然可以接收到返回的地址,但是原來函數(shù)棧幀存在的位置會被覆蓋,指針所指向的內(nèi)容會在此時被改變。所以說,return語句不可以返回指向棧內(nèi)存的指針,該函數(shù)棧幀在結(jié)束時即被銷毀。
到此這篇關(guān)于C語言從函數(shù)棧幀角度理解return關(guān)鍵字的文章就介紹到這了,更多相關(guān)C語言之return關(guān)鍵字內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C++11 模板參數(shù)的“右值引用”是轉(zhuǎn)發(fā)引用嗎
這篇文章主要介紹了C++11 模板參數(shù)的“右值引用”是轉(zhuǎn)發(fā)引用嗎,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-05-05
深入了解C++ 結(jié)構(gòu)體(struct)與共用體(union)
這篇文章主要介紹了C++ 結(jié)構(gòu)體與共用體的的相關(guān)資料,幫助大家更好的理解和學(xué)習(xí)c++,感興趣的朋友可以了解下2020-08-08
詳解Matlab繪制3D玫瑰花的方法(內(nèi)附旋轉(zhuǎn)版本)
這篇文章主要為大家介紹了如何利用Matlab繪制3D版的玫瑰花以及旋轉(zhuǎn)版的3D玫瑰花,文中的示例代碼講解詳細,感興趣的小伙伴可以動手試一試2022-03-03

