C++指向類成員函數(shù)的指針詳細(xì)解析
更新時間:2013年08月20日 09:14:40 作者:
由于這幾天在開發(fā)中要用到函數(shù)指針,所以就整理了一下關(guān)于函數(shù)指針的概念
首先 函數(shù)指針是指向一組同類型的函數(shù)的指針;而類成員函數(shù)我們也可以相似的認(rèn)為,它是指向同類中同一組類型的成員函數(shù)的指針,當(dāng)然這里的成員函數(shù)更準(zhǔn)確的講應(yīng)該是指非靜態(tài)的成員函數(shù)。前者是直接指向函數(shù)地址的,而后者我們從字面上也可以知道 它肯定是跟類和對象有著關(guān)系的。
函數(shù)指針實例:
復(fù)制代碼 代碼如下:
typedef int (*p)(int,int);//定義一個接受兩個int型且返回int型變量的函數(shù)指針類型
int func(int x,int y)
{
printf("func:x=%d,y=%d/n",x,y);
return (x<y?x:y);
}
int main()
{
p fun=func;//定義函數(shù)指針并給它賦上一個函數(shù)指針
cout<<"min:"<<(*fun)(4,5)<<endl;//為什么*fun需要用()擴起來呢?因為*的運算符優(yōu)先級比()低,如果不用()就成了*(fun())
return 0;
}
而“指向類成員函數(shù)的指針”卻多了一個類的區(qū)別:
class A
{
public:
int func(int x,int y)
{
printf("A::func:x=%d,y=%d/n",x,y);
return (x<y?x:y);
}
};
typedef int (A::*p)(int,int);//指針名前一定要加上所屬類型類名 A::的限定
int main()
{
p fun=&A::func;
A a; //因為成員函數(shù)地址的解引用必須要附駐與某個對象的地址,所以我們必須創(chuàng)建某個對象。
cout<<"min:"<<(a.*fun)(4,5)<<endl;
return 0;
}
嘿嘿。。只是用起來 .* 感覺怪怪滴。
接下來 我們可以再擴展一下下:
復(fù)制代碼 代碼如下:
#include <tchar.h>
#include <iostream>
#include <stdio.h>
using namespace std;
class A
{
public:
int func1(int x,int y)
{
printf("A::func:x=%d,y=%d/n",x,y);
return (x<y?x:y);
}
virtual int func2(int x,int y)
{
printf("A::func:x=%d,y=%d/n",x,y);
return (x>y?x:y);
}
};
class B:public A
{
public:
virtual int func2(int x,int y)
{
printf("B::func:x=%d,y=%d/n",x,y);
return (x+y);
}
};
typedef int (A::*p)(int,int);//指針名前一定要加上所屬類型類名 A::的限定
typedef int (B::*p0)(int,int);
int main()
{
A a; //因為成員函數(shù)地址的解引用必須要附駐與某個對象的地址,所以我們必須創(chuàng)建某個對象。
p fun=&A::func1;
cout<<(a.*fun)(4,5)<<endl;
cout<<(b.*fun)(4,5)<<endl<<endl;
fun=&A::func2;
cout<<(a.*fun)(4,5)<<endl;//請注意這里調(diào)用的是虛函數(shù),嘿嘿 還真神奇 類成員函數(shù)指針也支持多態(tài)。
cout<<(b.*fun)(4,5)<<endl<<endl;
//fun=&B::func2; //這樣式錯誤滴,因為不存在派生類的"指向類成員函數(shù)的指針"到基類的"指向類成員函數(shù)的指針"的隱式轉(zhuǎn)換
fun=(int (A::*)(int,int))&B::func2;//應(yīng)該進(jìn)行強制轉(zhuǎn)換
cout<<(a.*fun)(4,5)<<endl;
cout<<(b.*fun)(4,5)<<endl<<endl;
p0 fun0=&B::func2;
cout<<(a.*fun)(4,5)<<endl;
cout<<(b.*fun)(4,5)<<endl<<endl;
fun0=&A::func2; //正確,因為這里進(jìn)行了隱式轉(zhuǎn)換
cout<<(a.*fun)(4,5)<<endl;
cout<<(b.*fun)(4,5)<<endl<<endl;
//從上面我們不難發(fā)現(xiàn) 指向類成員函數(shù)的指針基類和派生類的關(guān)系和指向類對象的指針基類和派生類的關(guān)系完全相反,
//基類成員函數(shù)的布局被認(rèn)為是派生類成員函數(shù)布局的一個子集
return 0;
}
接下 是有關(guān)模板類的類成員函數(shù)指針的使用
實例如下:
復(fù)制代碼 代碼如下:
#include <tchar.h>
#include <iostream>
#include <stdio.h>
using namespace std;
class A
{
public:
int func(int x,int y)
{
printf("A::func : x=%d,y=%d/n",x,y);
return (x<y?x:y);
}
};
class B
{
public:
int func(int x,int y)
{
printf("B::func : x=%d,y=%d/n",x,y);
return (x>y?x:y);
}
};
template<class T>
class C
{
public:
T c;
void Print()
{
int (T::*p)(int,int)=&T::func;
(c.*p)(4,5);
}
};
int main()
{
C<A> ca;
C<B> cb;
ca.Print();
cb.Print();
return 0;
}
從上面 可以很清晰地看到。。其實它和普通的模板沒有什么區(qū)別。。只不過將限定名稱該為參數(shù)名酒OK啦。。。
嘿嘿。。。
相關(guān)文章
C語言植物大戰(zhàn)數(shù)據(jù)結(jié)構(gòu)二叉樹遞歸
這篇文章主要為大家介紹了C語言植物大戰(zhàn)數(shù)據(jù)結(jié)構(gòu)二叉樹遞歸,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-05-05
C++ 將一個文件讀入數(shù)組再讀出數(shù)組的方法
今天小編就為大家分享一篇C++ 將一個文件讀入數(shù)組再讀出數(shù)組的方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-07-07

