剖析C++編程當(dāng)中指針作為函數(shù)參數(shù)的用法
在C語言中,函數(shù)指針變量常見的用途之一是作為函數(shù)的參數(shù),將函數(shù)名傳給其他函數(shù)的形參。這樣就可以在調(diào)用一個函數(shù)的過程中根據(jù)給定的不同實參調(diào)用不同的函數(shù)。
例如,利用這種方法可以編寫一個求定積分的通用函數(shù),用它分別求5個函數(shù)的定積分:
可以看出,每次需要求定積分的函數(shù)是不一樣的??梢跃帉懸粋€求定積分的通用函數(shù)integral,它有3個形參: 下限a、上限b,以及指向函數(shù)的指針變量fun。函數(shù)原型可寫為:
double integral (double a, double b, double (*fun)(double));
分別編寫5個函數(shù)f1,f2,f3,f4,f5, 用來求上面5個函數(shù)的值。然后先后調(diào)用integral函數(shù)5次,每次調(diào)用時把a,b以及f1,f2,f3,f4,f5之一作為實參,即把上限、下限以及有關(guān)函數(shù)的入口地址傳送給形參fun。在執(zhí)行integral函數(shù)過程中求出各函數(shù)定積分的值。
在面向?qū)ο蟮腃++程序設(shè)計中,這種用法就比較少了。
函數(shù)的參數(shù)不僅可以是整型、浮點型、字符型等數(shù)據(jù),還可以是指針類型。它的作用是將一個變量的地址傳送給被調(diào)用函數(shù)的形參。
【例】即對輸入的兩個整數(shù)按大小順序輸出。這里用函數(shù)處理,而且用指針類型的數(shù)據(jù)作函數(shù)參數(shù)。程序如下:
#include <iostream> using namespace std; int main( ) { void swap(int *p1,int *p2); //函數(shù)聲明 int *pointer_1,*pointer_2,a,b; //定義指針變量pointer_1,pointer_2,整型變量a,b cin>>a>>b; pointer_1=&a; //使pointer_1指向a pointer_2=&b; //使pointer_2指向b if(a<b) swap(pointer_1,pointer_2); //如果a<b,使*pointer_1和*pointer_2互換 cout<<"max="<<a<<" min="<<b<<endl; //a已是大數(shù),b是小數(shù) return 0; } void swap(int *p1,int *p2) //函數(shù)的作用是將*p1的值與*p2的值交換 { int temp; temp=*p1; *p1=*p2; *p2=temp; }
運行情況如下:
45 78↙ max=78 min=45
請注意:不要將main函數(shù)中的swap函數(shù)調(diào)用寫成
if(a<b) swap(*pointer_1, *pointer_2);
請注意交換*p1和*p2的值是如何實現(xiàn)的。如果寫成以下這樣就有問題了:
void swap(int *p1, int *p2) { int *temp; *temp=*p1; //此語句有問題 *p1=*p2; *p2=*temp; }
本例采取的方法是交換a和b的值,而p1和p2的值不變。
可以看到,在執(zhí)行swap函數(shù)后,主函數(shù)中的變量a和b的值改變了。這個改變不是通過將形參值傳回實參來實現(xiàn)的。請讀者考慮一下能否通過調(diào)用下面的函數(shù)實現(xiàn)a和b互換。
void swap(int x, int y) { int temp; temp=x; x=y; y=temp; }
在main函數(shù)中用“swap(a, b);”調(diào)用swap函數(shù),會有什么結(jié)果呢?在函數(shù)調(diào)用時,a的值傳送給x,b的值傳送給y。執(zhí)行完swap函數(shù)最后一個語句后,x和y的值是互換了,但main函數(shù)中的a和b并未互換,如圖6.10(b)所示。也就是說由于虛實結(jié)合是采取單向的“值傳遞”方式,只能從實參向形參傳數(shù)據(jù),形參值的改變無法回傳給實參。
為了使在函數(shù)中改變了的變量值能被main函數(shù)所用,不能采取把要改變值的變量作為參數(shù)的辦法,而應(yīng)該用指針變量作為函數(shù)參數(shù)。在函數(shù)執(zhí)行過程中使指針變量所指向的變量值發(fā)生變化,函數(shù)調(diào)用結(jié)束后,這些變量值的變化依然保留下來,這樣就實現(xiàn)了“通過調(diào)用函數(shù)使變量的值發(fā)生變化,在主調(diào)函數(shù)中使用這些改變了的值”的目的。
如果想通過函數(shù)調(diào)用得到n個要改變的值,可以采取下面的步驟:
在主調(diào)函數(shù)中設(shè)n個變量,用n個指針變量指向它們;
編寫被調(diào)用函數(shù),其形參為n個指針變量,這些形參指針變量應(yīng)當(dāng)與主調(diào)函數(shù)中的n個指針變量具有相同的基類型;
在主調(diào)函數(shù)中將n個指針變量作實參,將它們的值(是地址值)傳給所調(diào)用函數(shù)的n個形參指針變量,這樣,形參指針變量也指向這n個變量;
通過形參指針變量的指向,改變該n個變量的值;
在主調(diào)函數(shù)中就可以使用這些改變了值的變量。
請注意,不能企圖通過改變形參指針變量的值而使實參指針變量的值改變。請分析下面程序:
#include <iostream> using namespace std; int main( ) { void swap(int *p1,int *p2); int *pointer_1,*pointer_2,a,b; cin>>a>>b; pointer_1=&a; pointer_2=&b; if(a<b) swap(pointer_1,pointer_2); cout<<"max="<<a<<" min="<<b<<endl; return 0; } void swap(int *p1,int *p2) { int *temp; temp=p1; p1=p2; p2=temp; }
實參變量和形參變量之間的數(shù)據(jù)傳遞是單向的“值傳遞”方式。指針變量作函數(shù)參數(shù)也要遵循這一規(guī)則。調(diào)用函數(shù)時不會改變實參指針變量的值,但可以改變實參指針變量所指向變量的值。
函數(shù)的調(diào)用可以(而且只可以)得到一個返回值(即函數(shù)值),而使用指針變量作函數(shù)參數(shù),就可以通過指針變量改變主調(diào)函數(shù)中變量的值,相當(dāng)于通過函數(shù)調(diào)用從被調(diào)用的函數(shù)中得到多個值。如果不用指針變量是難以做到這一點的。
【例】輸入a,b,c 3個整數(shù),按由大到小的順序輸出。
用上面介紹的方法,用3個指針變量指向3個整型變量,然后用swap函數(shù)來實現(xiàn)互換3個整型變量的值。程序如下:
#include <iostream> using namespace std; int main( ) { void exchange(int *,int *,int *); //對exchange函數(shù)的聲明 int a,b,c,*p1,*p2,*p3; cin>>a>>b>>c; //輸入3個整數(shù) p1=&a;p2=&b;p3=&c; //指向3個整型變量 exchange(p1,p2,p3); //交換p1,p2,p3指向的3個整型變量的值 cout<<a<<" "<<b<<" "<<c<<endl; //按由大到小的順序輸出3個整數(shù) } void exchange(int *q1,int *q2,int *q3) { void swap(int *,int *); //對swap函數(shù)的聲明 if(*q1<*q2) swap(q1,q2); //調(diào)用swap,將q1與q2所指向的變量的值互換 if(*q1<*q3) swap(q1,q3); //調(diào)用swap,將q1與q3所指向的變量的值互換 if(*q2<*q3) swap(q2,q3); //調(diào)用swap,將q2與q3所指向的變量的值互換 } void swap(int *pt1,int *pt2) //將pt1與pt2所指向的變量的值互換 { int temp; temp=*pt1; *pt1=*pt2; *pt2=temp; }
運行情況如下:
12 -56 87↙ 87 12 -56
- C++類的空指針調(diào)用成員函數(shù)的代碼
- c++11 符號修飾與函數(shù)簽名、函數(shù)指針、匿名函數(shù)、仿函數(shù)、std::function與std::bind
- C++中指針函數(shù)與函數(shù)指針的使用
- C++函數(shù)指針和回調(diào)函數(shù)使用解析
- C++根據(jù)傳入的函數(shù)指針來解析需要的參數(shù)(推薦)
- C++中回調(diào)函數(shù)及函數(shù)指針的實例詳解
- C++中函數(shù)指針詳解及代碼分享
- C++獲取類的成員函數(shù)的函數(shù)指針詳解及實例代碼
- 簡明的C++函數(shù)指針學(xué)習(xí)教程
- 實例解析C++中類的成員函數(shù)指針
- 解析C++中的字符串處理函數(shù)和指針
- c++ 函數(shù)指針相關(guān)總結(jié)
相關(guān)文章
C++實現(xiàn)LeetCode(190.顛倒二進制位)
這篇文章主要介紹了C++實現(xiàn)LeetCode(190.顛倒二進制位),本篇文章通過簡要的案例,講解了該項技術(shù)的了解與使用,以下就是詳細內(nèi)容,需要的朋友可以參考下2021-08-08構(gòu)造函數(shù)不能聲明為虛函數(shù)的原因及分析
構(gòu)造函數(shù)不需要是虛函數(shù),也不允許是虛函數(shù),因為創(chuàng)建一個對象時我們總是要明確指定對象的類型,盡管我們可能通過實驗室的基類的指針或引用去訪問它但析構(gòu)卻不一定,我們往往通過基類的指針來銷毀對象2013-10-10C語言中關(guān)于庫函數(shù) qsort 快排的用法
快速排序Qsort是所有學(xué)習(xí)算法和數(shù)據(jù)結(jié)構(gòu)最基礎(chǔ)的一個部分,也是考試題和面試的一個小重點。本片文章帶你了解Qsort的詳細用法規(guī)則2021-09-09