c++ 傳值,傳引用,傳指標比較
相關概念
c++三種傳遞引數方式 |
傳引數的值(稱值傳遞,簡稱傳值),傳引數的地址(稱地址傳遞,簡稱為傳址),和引用傳遞(稱為傳引用),相應的函式傳值呼叫,傳址呼叫和傳引用呼叫 |
形參、實參 |
函式定義時引數表中的引數稱為形式引數,簡稱形參;函式呼叫時引數表中的引數稱為實際引數,簡稱實參。實參和形參之間的資料傳遞稱為形實結合 |
說明 |
通常情況下,C++是傳值呼叫,它是單向的,只能從實參到形參。形參實質上市實參的一種拷貝,所以傳遞時不會改變外部實參的值 |
引用的含義及功能 |
即別名,功能主要是傳遞函式的引數和返回值 |
引用的規則 |
(1)引用被建立的同時必須被初始化(指標則可以在任何時候被初始化)。 (2)不能有NULL引用,引用必須與合法的儲存單元關聯(指標則可以是NULL)。 (3)一旦引用被初始化,就不能改變引用的關係(指標則可以隨時改變所指的物件)。 |
例項
#include <iostream>
using namespace std ;
void Swap(int x, int y) ;
int main(void)
{
int a = 1 ;
int b = 2 ;
cout << "a = " << a << ", " << "b = " << b << endl ;
Swap(a, b) ;
cout << "a = " << a << ", " << "b = " << b << endl ;
system("pause") ;
return 0 ;
}
一:值傳遞
void Swap(int x, int y)
{
int temp = x ;
x = y ;
y = temp ;
}
輸出結果:
a = 1, b = 2
a = 1, b = 2
原因:Swap(int x, int y)函式採用值傳遞的方式,傳入的實參實際上是a和b的副本而非其本身,所以對副本的改變並不會反應到a和b本身上。
二:引用傳遞
void Swap(int &x, int &y)
{
int temp = x ;
x = y ;
y = temp ;
}
輸出結果:
a = 1, b = 2
a = 2, b = 1
原因:Swap(int x, int y)函式採用引用傳遞的方式,傳入的實參實際上是a和b的引用,對引用的改變會直接反應到a和b本身上。
三:指標傳遞
1. 改變指標本身
void Swap(int *x, int *y)
{
int *temp = x ;
x = y ;
y = temp ;
}
呼叫方法:Swap(&a, &b) ;
輸出結果:
a = 1, b = 2
a = 1, b = 2
原因:Swap(int x, int y)函式採用指標傳遞的方式,傳入的實參實際上是a和b的指標的副本,而且改變的是副本本身而非其間接引用,所以不會影響的指標所指向的值,即a和b本身上。
2. 改變指標的間接引用
void Swap(int *x, int *y)
{
int temp = *x ;
*x = *y ;
*y = temp ;
}
呼叫方法:Swap(&a, &b) ;
輸出結果:
a = 1, b = 2
a = 2, b = 1
原因:Swap(int x, int y)函式採用指標傳遞的方式,傳入的實參雖然也是a和b的指標的副本,但是改變的是副本的間接引用,無論是指標本身還是其副本,都指向相同的值,所以這個改變會反應到a和b本身上。
效率比較
程式程式碼
//#include "stdafx.h"
#include "iostream"
#include "windows.h"
//#include "winbase.h"
using namespace std;
void TestValue(double val1,double val2);
void TestRef(double &val1,double &val2);
void TestPointer(double *val1,double *val2);
main()
{
double a = 1.5;
double b = 5.0;
DWORD dwStartTime;
DWORD dwEndTime;
dwStartTime= GetTickCount();
int i;
for(i=0;i<100000000;i++)
TestValue(a,b);
dwEndTime= GetTickCount();
cout << "\nTime1=" << dwEndTime - dwStartTime << endl;
dwStartTime= GetTickCount();
for(i=0;i<100000000;i++)
TestRef(a,b);
dwEndTime= GetTickCount();
cout << "\nTime2=" << dwEndTime - dwStartTime << endl;
dwStartTime= GetTickCount();
for(i=0;i<100000000;i++)
TestPointer(&a,&b);
dwEndTime= GetTickCount();
cout << "\nTime3=" << dwEndTime - dwStartTime << endl;
cin.get();
return 0;
}
void TestValue(double val1,double val2)
{
double val = val1 + val2;
}
void TestRef(double &val1,double &val2)
{
double val = val1 + val2;
}
void TestPointer(double *val1,double *val2)
{
double val = *val1 + *val2;
}
執行結果
Debug下:
第一次:1713 1718 1719
第二次:1714 1708 1709
第三次:1713 1718 1719
Release下:
第一次:266 266 218
第二次:282 265 219
第三次:266 265 235
可以進一步檢視反彙編程式碼,分析比較其指令
總結:傳引用=傳指標