java - C,C++函數調用時入棧順序 是編譯時還是運行時完成的?
問題描述
我的理解是靜態編譯時,因為C語言是過程性語言只能靜態聯編不能動態聯編,而C++編譯于此類似,所以是編譯時完全。不知道這樣對不對?
問題解答
回答1:入棧的順序是編譯時確定的。
函數調用之前需要入棧的主要是函數參數,而參數都是固定的(可變參數只是用宏確定偏移量)。 調用函數的代碼是放在代碼段的,入棧都是以指令方式進行的,所以順序都是編譯時確定的。
回答2:@lianera 說的不錯,入棧的順序是編譯時確定的。
我這給你看個例子:我有段代碼是這樣的
#include <stdio.h>int test_fun(int a, int b){ return a + b; }int main(int argc, char *argv[]){ int A, B, ret; A = 3; B = 4; ret = test_fun(A, B); return 1;}
編譯后,他的匯編代碼是這樣的
int test_fun(int a, int b){ 400474: 55 push %rbp 400475: 48 89 e5 mov %rsp,%rbp // $edi存的是A的值,$esi存的是B的值,將他們壓入棧中 400478: 89 7d fc mov %edi,-0x4(%rbp) 40047b: 89 75 f8 mov %esi,-0x8(%rbp) return a + b; 40047e: 8b 45 f8 mov -0x8(%rbp),%eax 400481: 8b 55 fc mov -0x4(%rbp),%edx 400484: 8d 04 02 lea (%rdx,%rax,1),%eax}int main(int argc, char *argv[]){ 400489: 55 push %rbp 40048a: 48 89 e5 mov %rsp,%rbp 40048d: 48 83 ec 20 sub $0x20,%rsp 400491: 89 7d ec mov %edi,-0x14(%rbp) 400494: 48 89 75 e0 mov %rsi,-0x20(%rbp) int A, B, ret; // 壓入本地變量A A = 3; 400498: c7 45 f4 03 00 00 00 movl $0x3,-0xc(%rbp) // 壓入本地變量B B = 4; 40049f: c7 45 f8 04 00 00 00 movl $0x4,-0x8(%rbp) ret = test_fun(A, B); 4004a6: 8b 55 f8 mov -0x8(%rbp),%edx 4004a9: 8b 45 f4 mov -0xc(%rbp),%eax // 將A和B的值放入相應的寄存器 4004ac: 89 d6 mov %edx,%esi 4004ae: 89 c7 mov %eax,%edi // 調用test_fun 4004b0: e8 bf ff ff ff callq 400474 <test_fun> 4004b5: 89 45 fc mov %eax,-0x4(%rbp) return 1; 4004b8: b8 01 00 00 00 mov $0x1,%eax}
不懂匯編也沒關系,在編譯過程中,參數的傳遞順序,參數、本地變量等應該放在棧的哪個位置(相對位置)都是定了的。當程序運行到相應程序后會按照編譯好的順序對棧進行操作。
回答3:入棧這個不是運行時才有的過程嗎?編譯只是翻譯為字節碼的過程,為什么會有入棧?
相關文章:
1. java - C++ NULL賦值給指針2. java - C++ ostream的成員函如put(),str()還有哪些?哪里可查?3. java - Cannot resolve symbol ‘HttpServletRequest’4. java - Cas操作會加鎖嗎5. java - c++ 數據結構這個錯在哪了,動態存儲?6. java - c語言 數據結構 順序存儲結構線性表初始化問題7. java - c3p0連接池配置問題8. java - Collections類里的swap函數,源碼為什么要新定義一個final的List型變量l指向傳入的list?9. java - CSV數據格式引號內包含分隔符如何寫正則表達式10. java - controller返回的String,并沒有這個地址啊?
