當(dāng)你在函數(shù)的最后寫(xiě)上 return 0 的時(shí)候,它是如何返回給調(diào)用函數(shù)的?
比如 test 函數(shù),為了待會(huì)更好的看懂匯編代碼,我寫(xiě)成了 return 1234。
處理函數(shù)的返回值,是不是像我們理解的那樣,直接把 1234 賦值給了變量 ret?
搞懂這個(gè)問(wèn)題不難,只要看下匯編代碼就行。
把代碼編譯一下,只編譯不鏈接,得到的就是C對(duì)應(yīng)的匯編代碼。
這塊是 test 函數(shù),不用管上面這些代碼,如果一行一行去分析,沒(méi)有匯編基礎(chǔ)的話確實(shí)會(huì)頭疼。
看下這行代碼,很明顯,1234 就是我剛才寫(xiě)的返回值。所以 return 1234,其實(shí)就是把 1234 放到了寄存器 EAX 中。
EAX 是 X86 架構(gòu)下的 32 位寄存器,在這個(gè)地方用于保存函數(shù)的返回值。
在回到主函數(shù),通過(guò) call 指令調(diào)用了 test 函數(shù),緊接這就把 EAX 寄存器的值放到了 RBP 寄存器減 4 個(gè)字節(jié)的地址處,這個(gè)地址就是局部變量 ret 的地址。
所以這個(gè)過(guò)程非常簡(jiǎn)單,test 函數(shù)把返回值 1234 放到寄存器 EAX 中,主函數(shù)再?gòu)?EAX 把數(shù)據(jù)讀到 ret 中。
把代碼修改下,如果返回的是指針,指針占 8 個(gè)字節(jié),匯編代碼中也只是把 EAX 寄存器換成了 RAX 寄存器,這是一個(gè) 64 位的寄存器,剛好可以存放 8 個(gè)字節(jié)的指針。
不管函數(shù)返回什么類(lèi)型,char short int long 或者指針,都可以通過(guò)這兩個(gè)寄存器來(lái)完成。
于是又有了新的問(wèn)題,如果返回結(jié)構(gòu)體怎么辦?結(jié)構(gòu)體的大小可能遠(yuǎn)遠(yuǎn)超過(guò) 8 個(gè)字節(jié)。
之前我們也講過(guò)這個(gè)問(wèn)題,不同的編譯器處理方法可能不一樣。
比如我用的環(huán)境,調(diào)用函數(shù)之前,把局部變量 ret 的地址作為參數(shù)傳給了 test 函數(shù),實(shí)際上,我們?cè)趯?xiě)代碼的時(shí)候,test并沒(méi)有參數(shù)。最終返回結(jié)構(gòu)體,其實(shí)通過(guò)傳進(jìn)來(lái)的指針,把結(jié)構(gòu)體的內(nèi)容復(fù)制到了變量 ret 里面。
-
C語(yǔ)言
+關(guān)注
關(guān)注
180文章
7630瀏覽量
140200 -
函數(shù)
+關(guān)注
關(guān)注
3文章
4367瀏覽量
64155
原文標(biāo)題:C語(yǔ)言如何處理函數(shù)的返回值
文章出處:【微信號(hào):學(xué)益得智能硬件,微信公眾號(hào):學(xué)益得智能硬件】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。
發(fā)布評(píng)論請(qǐng)先 登錄
請(qǐng)問(wèn)CY68013上位機(jī)發(fā)送的返回值到底代表什么意思?
如何把兩個(gè)數(shù)據(jù)返回給調(diào)用函數(shù)
EE-128:C語(yǔ)言中的DSP:從C調(diào)用匯編類(lèi)成員函數(shù)

ADS1115返回值不連續(xù)是怎么回事?
ADS8332數(shù)據(jù)不刷新,每個(gè)通道返回值都是1.7V左右,而且會(huì)慢慢上升,為什么?
同樣是函數(shù),在C和C++中有什么區(qū)別
常用SQL函數(shù)及其用法
使用C語(yǔ)言實(shí)現(xiàn)函數(shù)模板
HTTP相關(guān)返回值異常如何解決(下篇)

HTTP相關(guān)返回值異常如何解決(上篇)

面試常考+1:函數(shù)指針與指針函數(shù)、數(shù)組指針與指針數(shù)組

評(píng)論