1. 程式人生 > >c++ 傳值,傳引用,傳指標比較

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

可以進一步檢視反彙編程式碼,分析比較其指令

總結:傳引用=傳指標