您好,登錄后才能下訂單哦!
本文小編為大家詳細(xì)介紹“C語言非void函數(shù)卻沒有return會(huì)有什么影響”,內(nèi)容詳細(xì),步驟清晰,細(xì)節(jié)處理妥當(dāng),希望這篇“C語言非void函數(shù)卻沒有return會(huì)有什么影響”文章能幫助大家解決疑惑,下面跟著小編的思路慢慢深入,一起來學(xué)習(xí)新知識(shí)吧。
前幾天學(xué)習(xí)棧, 寫了一個(gè)創(chuàng)建棧的函數(shù)
typedef struct{ int data[STACKSIZE]; int top; }stack, *Stack;Stack NewStack(void){ Stack s = (Stack)malloc(sizeof(stack)); s->top = -1; }
代碼的作用很簡(jiǎn)單, 就是動(dòng)態(tài)分配一個(gè)棧變量的內(nèi)存, 并將其指針返回; 很顯然的是在函數(shù)NewStack中少了一句return s;
。
令人奇怪的是我直到最后才偶然發(fā)現(xiàn)這個(gè)錯(cuò)誤, 因?yàn)檫@個(gè)函數(shù)在整個(gè)代碼運(yùn)行中都是正常的, 并沒有發(fā)生錯(cuò)誤。
這里先下一個(gè)結(jié)論, NewStack正確的將s的值返回了, 具體為什么先按下不表。
再來看這樣一個(gè)函數(shù)
int max(int x, int y){ int max; if(x >= y){ max = x; } else{ max = y; } }
代碼的作用一目了然, 但缺失了return max;
, 這個(gè)函數(shù)返回什么呢?
運(yùn)行結(jié)果如下
兩次函數(shù)調(diào)用都返回了正確的最大值, 為什么呢?
莫非是編譯器已經(jīng)智能到能猜出用戶的想法, 或是因?yàn)閙ax是該函數(shù)棧中唯一一個(gè)用戶定義的變量, 總之, 是不是max被返回了呢?
int max(int x, int y){ int max; if(x >= y){ max = x; } else{ max = y; } max = 1000; }
我們?cè)谀┪布由弦痪鋗ax = 1000;, 驗(yàn)證一下返回值究竟是不是max的值。
運(yùn)行結(jié)果如下
顯然這里并不是簡(jiǎn)單的返回了max的值。
我們?cè)僭谀┪布右痪?code>max = x時(shí)
運(yùn)行結(jié)果如下
可以看到返回值始終是x的值。
所以
max = 1000;
——沒有改變返回值
max = x;
——改變了返回值
讓我們來看看stackflow上的一段解答
大意就是在某些系統(tǒng)、某些編譯器下(windows, gcc), 如果一個(gè)非void函數(shù)沒有返回值, 系統(tǒng)會(huì)自動(dòng)幫你返回eax
寄存器中的值, 在這里表現(xiàn)為最后一個(gè)表達(dá)式的值。
對(duì)max函數(shù)進(jìn)行反匯編,
可以發(fā)現(xiàn), 在諸如x >= y, max = x:這樣的涉及至少兩個(gè)變量的語句都使用了eax寄存器, 因而改變的eax寄存器中的值;而max = 1000則沒有, 因?yàn)閙ax = 1000只對(duì)一塊內(nèi)存進(jìn)行操作, 根本不需要用到寄存器。
諸如max = x;這樣的語句, 內(nèi)存中的實(shí)現(xiàn)大概就是, 將x的值移動(dòng)到eax寄存器, 再將eax中的值移動(dòng)到max(因?yàn)榧拇嫫鞯拇嫒∷俣冗h(yuǎn)高于內(nèi)存)。
除此之外, eax寄存器也會(huì)用在存儲(chǔ)函數(shù)的返回值上。
如我們?cè)诤瘮?shù)末尾加一個(gè)函數(shù)調(diào)用。
返回值始終是max函數(shù)內(nèi)調(diào)用的f函數(shù)的返回值。
在windows、gcc環(huán)境下, 若非void函數(shù)無返回值, 系統(tǒng)會(huì)自動(dòng)的返回當(dāng)時(shí)eax寄存器中的值, 通常表現(xiàn)為函數(shù)中最后一條語句(兩個(gè)變量以上)或函數(shù)調(diào)用的值。在其它的C編譯器下, 返回值可能會(huì)是某個(gè)固定的值, 也可能是隨機(jī)值。
C編譯器對(duì)這一問題的處理能很好的反映C對(duì)用戶的信任與包容, 在其它語言上這種情況一般是會(huì)報(bào)錯(cuò)的, 而在C中甚至連警告都沒有。C總是傾向于認(rèn)為用戶的做法有他自己的道理, 但更多情況下沒人會(huì)去利用如此的特性, 只是單純的疏忽。
那我們?cè)撛趺慈ケ苊膺@樣的錯(cuò)誤呢?
只需要在編譯器選項(xiàng)中加上一條指令Wreturn-type
, 再發(fā)生這種情況的時(shí)候gcc編譯器就會(huì)給我們一個(gè)提醒了。
讀到這里,這篇“C語言非void函數(shù)卻沒有return會(huì)有什么影響”文章已經(jīng)介紹完畢,想要掌握這篇文章的知識(shí)點(diǎn)還需要大家自己動(dòng)手實(shí)踐使用過才能領(lǐng)會(huì),如果想了解更多相關(guān)內(nèi)容的文章,歡迎關(guān)注億速云行業(yè)資訊頻道。
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。