關於指標與參照傳遞的效率問題

2023-08-30 21:00:26

引言

  • 參照是C++的特性,指標是C語言的特性
  • 關於這兩種特性的執行效率,人云亦云,好多人都說參照傳遞效率更高
  • 以至於一些面試官在自己都不清楚的前提下面試別人
  • 筆者有幸遇到過,由於看過底層組合,在面試官對我說參照效率更高的時候,導致我一度懷疑自己的記憶力
  • 下面我們就看看參照在組合層面與指標有什麼區別吧

DEMO(main.cpp)

#include <iostream>
#include <cstring>
void t1(int &b)
{
    ++b;
    return;
}

void t2(int *c)
{
    ++*c;
    return;
}
int main()
{
    int a = 100;
    t1(a);
    t2(&a);
    return 0;
}

編譯

g++ -g -o test ./main.cpp

反編譯

objdump -S ./test > ./test.S

AT&T(test.S)

  • 由於是c++程式碼,所以組合檔案比較大
  • 為了方便閱讀,此處僅摘抄重點部分
00000000000007aa <_Z2t1Ri>:
#include <iostream>
#include <cstring>
void t1(int &b)
{
 7aa:	55                   	push   %rbp
 7ab:	48 89 e5             	mov    %rsp,%rbp
 7ae:	48 89 7d f8          	mov    %rdi,-0x8(%rbp)
    ++b;
 7b2:	48 8b 45 f8          	mov    -0x8(%rbp),%rax
 7b6:	8b 00                	mov    (%rax),%eax
 7b8:	8d 50 01             	lea    0x1(%rax),%edx
 7bb:	48 8b 45 f8          	mov    -0x8(%rbp),%rax
 7bf:	89 10                	mov    %edx,(%rax)
    return;
 7c1:	90                   	nop
}
 7c2:	5d                   	pop    %rbp
 7c3:	c3                   	retq   

00000000000007c4 <_Z2t2Pi>:

void t2(int *c)
{
 7c4:	55                   	push   %rbp
 7c5:	48 89 e5             	mov    %rsp,%rbp
 7c8:	48 89 7d f8          	mov    %rdi,-0x8(%rbp)
    ++*c;
 7cc:	48 8b 45 f8          	mov    -0x8(%rbp),%rax
 7d0:	8b 00                	mov    (%rax),%eax
 7d2:	8d 50 01             	lea    0x1(%rax),%edx
 7d5:	48 8b 45 f8          	mov    -0x8(%rbp),%rax
 7d9:	89 10                	mov    %edx,(%rax)
    return;
 7db:	90                   	nop
}
 7dc:	5d                   	pop    %rbp
 7dd:	c3                   	retq   

00000000000007de <main>:
int main()
{
 7de:	55                   	push   %rbp
 7df:	48 89 e5             	mov    %rsp,%rbp
 7e2:	48 83 ec 10          	sub    $0x10,%rsp
 7e6:	64 48 8b 04 25 28 00 	mov    %fs:0x28,%rax
 7ed:	00 00 
 7ef:	48 89 45 f8          	mov    %rax,-0x8(%rbp)
 7f3:	31 c0                	xor    %eax,%eax
    int a = 100;
 7f5:	c7 45 f4 64 00 00 00 	movl   $0x64,-0xc(%rbp)
    t1(a);
 7fc:	48 8d 45 f4          	lea    -0xc(%rbp),%rax
 800:	48 89 c7             	mov    %rax,%rdi
 803:	e8 a2 ff ff ff       	callq  7aa <_Z2t1Ri>
    t2(&a);
 808:	48 8d 45 f4          	lea    -0xc(%rbp),%rax
 80c:	48 89 c7             	mov    %rax,%rdi
 80f:	e8 b0 ff ff ff       	callq  7c4 <_Z2t2Pi>
    return 0;
 814:	b8 00 00 00 00       	mov    $0x0,%eax
}
 819:	48 8b 55 f8          	mov    -0x8(%rbp),%rdx
 81d:	64 48 33 14 25 28 00 	xor    %fs:0x28,%rdx
 824:	00 00 
 826:	74 05                	je     82d <main+0x4f>
 828:	e8 43 fe ff ff       	callq  670 <__stack_chk_fail@plt>
 82d:	c9                   	leaveq 
 82e:	c3                   	retq   

初步結論

  • 我們通過編譯與反組合可以看到
  • 不論指標還是參照,所有組合程式碼除了t1,t2地址的不同,可以說沒有任何區別
  • 故參照其實就是指標,不過是c++幫你解除參照(加了*號)並進行了一定的語法限制
  • 以上組合中或許有一些我沒注意到的細節,歡迎各位大佬在評論區指出

完善DEMO

#include <iostream>
#include <cstring>
void t1(int &b)
{
    ++b;
    return;
}

void t2(int *c)
{
    ++*c;
    return;
}
int main(int argc,char **argv)
{
    int a = 100;
    long b = 10000000000;

    bool ptr = false;
    if(argc > 1 && strstr(argv[1],"p")) ptr = true;

    if(!ptr) 	while(--b) t1(a); 
    else		while(--b) t2(&a);

    return 0;
}

比對執行效率

  • 考慮到環境因素帶來的不確定性,比如cpu降頻,其它程序搶佔cpu等
  • 故我此處執行了多次,其中帶有引數p的是使用的指標,不帶有任何引數的是使用的參照
kbin@kbin-virtual-machine:~/test$ time ./test 
	real	0m18.444s
	user	0m18.391s
	sys	0m0.036s
kbin@kbin-virtual-machine:~/test$ time ./test p
	real	0m18.173s
	user	0m18.141s
	sys	0m0.016s
kbin@kbin-virtual-machine:~/test$ time ./test 
	real	0m18.424s
	user	0m18.418s
	sys	0m0.000s
kbin@kbin-virtual-machine:~/test$ time ./test p
	real	0m18.261s
	user	0m18.156s
	sys	0m0.088s
kbin@kbin-virtual-machine:~/test$ time ./test
	real	0m18.470s
	user	0m18.429s
	sys	0m0.028s
kbin@kbin-virtual-machine:~/test$ time ./test p
	real	0m18.300s
	user	0m18.282s
	sys	0m0.008s
kbin@kbin-virtual-machine:~/test$ time ./test
	real	0m18.434s
	user	0m18.402s
	sys	0m0.028s
kbin@kbin-virtual-machine:~/test$ time ./test p
	real	0m18.283s
	user	0m18.259s
	sys	0m0.008s
  • 可以看到指標甚至在效率上高於參照
  • 當然這是由於誤差導致的...

最終結論

  • 指標與參照在執行效率上是不分伯仲的
  • 喜歡用指標還是參照完全憑藉個人喜好
  • 指標在使用的靈活度上具有很高的優勢,但如果使用過程中不注意細節,就會存在安全隱患
  • 參照由於受到c++語法的限制,犧牲了一定的靈活性,但卻大大提高了使用過程中的安全性
  • 至於網路上說參照更具有執行效率,或許是因為指標在使用前一般會去判斷非NULL吧...