C++引用原理

1、直接定義引用變量

示例代碼

#include<iostream>

void fun() {
	int a = 1;
	int& b = a;
	b = 10;
}

int main() {
	fun();
	return 0;
}

代碼反彙編結果(使用VS反彙編,只看fun函數的代碼即可)

#include<iostream>

void fun() {
00007FF7AAD01920  push        rdi  
00007FF7AAD01922  sub         rsp,40h  
00007FF7AAD01926  mov         rdi,rsp  
00007FF7AAD01929  mov         ecx,10h  
00007FF7AAD0192E  mov         eax,0CCCCCCCCh  
00007FF7AAD01933  rep stos    dword ptr [rdi]  

// 對變量a進行定義,a的地址爲 rsp + 24h, 把這個地址的內容初始化爲1
	int a = 1;
00007FF7AAD01935  mov         dword ptr [rsp+24h],1  
// 創建變量b,b是a的引用,b 是指向 a 的指針,變量b的地址爲 rsp+38h
// lea         rax,[rsp+24h]   取得變量a的地址,並把地址放到rax寄存器中
// qword ptr [rsp+38h],rax 把rax的內容賦給變量b
	int& b = a;
00007FF7AAD0193D  lea         rax,[rsp+24h]  
00007FF7AAD01942  mov         qword ptr [rsp+38h],rax  

// 修改變量b的值的時候,首先取得b中的內容到寄存器中,然後以寄存器中的內容作爲地址,修改對應地址的數據
// mov         rax,qword ptr [rsp+38h]  把 b 的值mov到rax寄存器中
// mov         dword ptr [rax],0Ah 把地址爲rax的內存內容設置爲 0AH(10)
	b = 10;
00007FF7AAD01947  mov         rax,qword ptr [rsp+38h]  
00007FF7AAD0194C  mov         dword ptr [rax],0Ah  
}
00007FF7AAD01952  mov         rcx,rsp  
00007FF7AAD01955  lea         rdx,[00007FF7AAD09E30h]  
00007FF7AAD0195C  call        00007FF7AAD01221  
00007FF7AAD01961  add         rsp,40h  
00007FF7AAD01965  pop         rdi  
00007FF7AAD01966  ret  

2、方法參數傳引用

示例代碼

#include<iostream>

void fun(int &a) {
	a = 10;
}

int main() {
	int a = 1;
	fun(a);

	std::cout << a << std::endl;
	return 0;
}

代碼反彙編結果(從main函數看起)

#include<iostream>

void fun(int &a) {
// 把 rsp+8 地址的內容設置爲寄存器 rcx 中的內容(rcx爲請求參數,內容爲main函數中定義的變量a的地址)
00007FF746531470  mov         qword ptr [rsp+8],rcx  
00007FF746531475  push        rdi  
// 執行代碼 a = 10;  首先取得 rsp+10h 中的值,然後以 rsp+10h 中的值作爲地址修改對應內存中的數據
	a = 10;
00007FF746531476  mov         rax,qword ptr [rsp+10h]  
00007FF74653147B  mov         dword ptr [rax],0Ah  
}
00007FF746531481  pop         rdi  
00007FF746531482  ret  

// 中間省略一堆垃圾代碼

int main() {
00007FF746531490  push        rdi  
00007FF746531492  sub         rsp,40h  
00007FF746531496  mov         rdi,rsp  
00007FF746531499  mov         ecx,10h  
00007FF74653149E  mov         eax,0CCCCCCCCh  
00007FF7465314A3  rep stos    dword ptr [rdi]  
// 定義變量a,並把a的值初始化爲1, a的地址爲 rsp+24h
	int a = 1;
00007FF7465314A5  mov         dword ptr [rsp+24h],1  
// 調用 fun函數,由於參數較少,因此此函數調用使用寄存器傳參
// lea         rcx,[rsp+24h]  把rcx寄存器的值設置爲參數的a的地址
// 使用call指令調用 fun 函數
	fun(a);
00007FF7465314AD  lea         rcx,[rsp+24h]  
00007FF7465314B2  call        00007FF74653107D  
	std::cout << a << std::endl;
00007FF7465314B7  mov         edx,dword ptr [rsp+24h]  
00007FF7465314BB  mov         rcx,qword ptr [00007FF74653E170h]  
00007FF7465314C2  call        qword ptr [00007FF74653E158h]  
00007FF7465314C8  lea         rdx,[00007FF74653108Ch]  
00007FF7465314CF  mov         rcx,rax  
00007FF7465314D2  call        qword ptr [00007FF74653E160h]  
	return 0;
00007FF7465314D8  xor         eax,eax  
}
00007FF7465314DA  mov         edi,eax  
00007FF7465314DC  mov         rcx,rsp  
00007FF7465314DF  lea         rdx,[00007FF746539E40h]  
00007FF7465314E6  call        00007FF7465311F9  
00007FF7465314EB  mov         eax,edi  
00007FF7465314ED  add         rsp,40h  
00007FF7465314F1  pop         rdi  
00007FF7465314F2  ret  

總結

C ++ 中的引用和指針並無本質區別,只是編譯器自動對指針做了取值操作而已

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章