亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > C++ > 正文

實例解析C++中類的成員函數指針

2020-01-26 14:38:24
字體:
來源:轉載
供稿:網友

C語言的指針相當的靈活方便,但也相當容易出錯。許多C語言初學者,甚至C語言老鳥都很容易栽倒在C語言的指針下。但不可否認的是,指針在C語言中的位置極其重要,也許可以偏激一點的來說:沒有指針的C程序不是真正的C程序。
然而C++的指針卻常常給我一種束手束腳的感覺。C++比C語言有更嚴格的靜態類型,更加強調類型安全,強調編譯時檢查。因此,對于C語言中最容易錯用的指針,更是不能放過:C++的指針被分成數據指針,數據成員指針,函數指針,成員函數指針,而且不能隨便相互轉換。而且這些指針的聲明格式都不一樣:

數據指針 T *
成員數據指針 T::*
函數指針 R (*)(...)
成員函數指針 R (T::*)(...)

還有一個更重要的區別是,指針所占的空間也不一樣了。即使在32位系統中,所占的空間也有可能是4字節、8字節、12字節甚至16字節,這個依據平臺及編譯器,有很大的變化。
盡管C++中仍然有萬能指針void*,但它卻屬于被批斗的對象,而且再也不能“萬能”了。它不能轉換成成員指針。
這樣一來,C++的指針就變得很尷尬:我們需要一種指針能夠指向同一類型的數據,不管這個數據是普通數據,還是成員數據;我們更需要一種指針能夠指向同一類型的函數,不管這個函數是靜態函數,還是成員函數。但是沒有,至少從現在的C++標準中,還沒有看到。
在編程工作中常會遇到在一個“類”中通過函數指針調用成員函數的要求,如,當在一個類中使用了C++標準庫中的排序函數qsort時,因qsort參數需要一個“比較函數”指針,如果這個“類”使用某個成員函數作“比較函數”,就需要將這個成員函數的指針傳給qsort供其調用。本文所討論的用指針調用 “類”的成員函數包括以下三種情況:

(1).將 “類”的成員函數指針賦予同類型非成員函數指針,如:

例子1

#include <stdlib.h> typedef void (*Function1)( ); //定義一個函數指針類型。 Function1 f1; class Test1 {  public:  //…被調用的成員函數。 void Memberfun1( ){ printf("%s /n","Calling Test3::Memberfun2 OK");}; //  void Memberfun2() {  f1=reinterpret_cast<Function1>(Memberfun1);//將成員函數指針賦予f1。編譯出錯。  f1(); } //… }; int main() {  Test1 t1;  t1.Memberfun2();  return 0; } 

(2) 在一個“類”內,有標準庫函數,如qsort, 或其他全局函數,用函數指針調用類的成員函數。如:

例子2:

#include <stdlib.h> class Test2 { private:  int data[2];  //… public: //… int __cdecl Compare(const void* elem1, const void* elem2) //成員函數。 {  printf("%s /n","Calling Test2::Memberfun OK"); return *((int*)elem1)- *((int*)elem2) ;  } void Memberfun()  {  data[0]=2; data[1]=5; qsort( data, 2, sizeof(int), Compare); //標準庫函數調用成 //員函數。編譯出錯。 } //… }; int main( ) { Test2 t2; t2.Memberfun(); //調用成員函數。 return 0; } 

(3)同一個“類”內,一個成員函數調用另一個成員函數, 如:

例子3:

#include "stdlib.h" class Test3 { public: //… void Memberfun1( void (* f2)( ) ) { f2( ) ;} //成員函數1調用成員函數//2。 void Memberfun2( ) { printf("%s /n","Calling Test3::Memberfun2 OK");} //成員函數2。 void Memberfun3( ) { Memberfun1( Memberfun2);} // 編譯出錯  //… }; int main( ) { Test3 t3; t3.Memberfun3(); //調用成員函數。 return 0; } 

以上三種情況的代碼語法上沒有顯著的錯誤,在一些較早的編譯環境中,如,VC++ 4.0, 通??梢跃幾g通過,或至多給出問題提醒(Warning)。后來的編譯工具,如,VC++6.0和其他一些常用的C++編譯軟件,不能通過以上代碼的編譯, 并指出錯誤如下(以第三種情況用VC++ 6.0編譯為例):

error C2664: 'Memberfun1' : cannot convert parameter 1 from 'void (void)' to 'void (__cdecl *)(void)'None of the functions with this name in scope match the target type

即:Memberfun1參數中所調用的函數類型不對。

按照以上提示,僅通過改變函數的類型無法消除錯誤,但是,如果單將這幾個函數從類的定義中拿出來,不作任何改變就可以消除錯誤通過編譯, 仍以第三種情況為例,以下代碼可通過編譯:

#include <stdlib.h> void Memberfun1( void (* f2)( ) ) { f2( ) ;} //原成員函數1調用成員函數//2。 void Memberfun2( ) { printf("%s /n","Calling Test3::Memberfun2 OK");} //原成員函數2。 void Memberfun3( ) { Memberfun1( Memberfun2);} int main( ) { Memberfun3 (); return 0; } 

第1、 2種情況和第3種情況完全相同。

由此可以的得出結論,以上三種情況編譯不能通過的原因表面上并不在于函數類型調用不對,而是與 “類”有關。沒通過編譯的情況是用函數指針調用了 “類”的成員函數,通過編譯的是用函數指針調用了非成員函數,而函數的類型完全相同。那么, “類”的成員函數指針和非成員函數指針有什么不同嗎?

在下面的程序中,用sizeof()函數可以查看各種“類”的成員函數指針和非成員函數指針的長度(size)并輸出到屏幕上。

#include "stdafx.h" #include <iostream> #include <typeinfo.h> class Test; //一個未定義的類。 class Test2 //一個空類。 { }; class Test3 //一個有定義的類。 {  public: //... void (* memberfun)(); void Memberfun1( void (* f2)( ) ) { f2( ) ;} //成員函數1調用成員函數//2。 void Memberfun2( );//成員函數2。 //… }; class Test4: virtual Test3 ,Test2 //一個有virtual繼承的類(derivative class)。 {  public: void Memberfun1( void (* f2)( ) ) { f2( ) ;}  }; class Test5: Test3,Test2 //一個繼承類(derivative class)。 {  public: void Memberfun1( void (* f2)( ) ) { f2( ) ;}  };  int main() {  std::cout <<"一般函數指針長度= "<< sizeof(void(*)()) << '/n';  std::cout <<"-類的成員函數指針長度-"<<'/n'<<'/n';  std::cout <<"Test3類成員函數指針長度="<< sizeof(void(Test3::*)())<<'/n'<<'/n';  std::cout <<"Test5類成員函數指針長度="<<sizeof(void (Test5:: *)())<<'/n';  std::cout <<"Test4類成員函數指針長度="<<sizeof(void (Test4:: *)())<<'/n';  std::cout <<"Test類成員函數指針長度="<<sizeof(void(Test::*)()) <<'/n';  return 0; } 

輸出結果為(VC++6.0編譯,運行于Win98操作系統,其他操作系統可能有所不同):

  • 一般非成員函數指針長度= 4
  • -類的成員函數指針長度-
  • Test3類成員函數指針長度=4
  • Test5類成員函數指針長度=8
  • Test4類成員函數指針長度=12
  • Test類成員函數指針長度=16

以上結果表明,在32位Win98操作系統中,一般函數指針的長度為4個字節(32位),而類的成員函數指針的長度隨類的定義與否、類的繼承種類和關系而變,從無繼承關系類(Test3)的4字節(32位)到有虛繼承關系類(Virtual Inheritance)(Test4)的12字節(96位),僅有說明(declaration)沒有定義的類(Test)因為與其有關的一些信息不明確成員函數指針最長為16字節(128位)。顯然, 與一般函數指針不同,指向“類”的成員函數的指針不僅包含成員函數地址的信息,而且包含與類的屬性有關的信息,因此,一般函數指針和類的成員函數指針是根本不同的兩種類型,當然,也就不能用一般函數指針直接調用類的成員函數,這就是為什么本文開始提到的三種情況編譯出錯的原因。盡管使用較早版本的編譯軟件編譯仍然可以通過,但這會給程序留下嚴重的隱患。

至于為什么同樣是指向類的成員函數的指針,其長度竟然不同,從32位到128位,差別很大,由于沒有看到微軟官方的資料只能推測VC++6.0在編譯時對類的成員函數指針進行了優化,以盡量縮短指針長度,畢竟使用128位或96位指針在32位操作系統上對程序性能會有影響。但是,無論如何優化,類的成員函數指針包含一定量的對象(Objects)信息是確定的。其他的操作系統和編譯軟件是否進行了類似的處理,讀者可以用以上程序自己驗證。

那么,當需要時,如何用指針調用類的成員函數?可以考慮以下方法:

(1) 將需要調用的成員函數設為static 類型,如:在前述例子2中,將class Test2 成員函數Compare 定義前加上static 如下:

class Test2 { //…. int static __cdecl Compare(const void* elem1, const void* elem2) //成員函數。 //其他不變 } 

改變后的代碼編譯順利通過。原因是,static 類型的成員函數與類是分開的,其函數指針也不包含對象信息,與一般函數指針一致。這種方法雖然簡便,但有兩個缺點:1、被調用的函數成員定義內不能出現任何類的成員(包括變量和函數);2、由于使用了static 成員,類在被繼承時受到了限制。

(2) 使用一個函數參數含有對象信息的static 類型的成員函數為中轉間接地調用其他成員函數,以例3為例,將類Test3作如下修改,main()函數不變,則可順利通過編譯:

class Test3 {  public: //… void static __cdecl Helper(Test3* test3) {  test3->Memberfun2(); } void Memberfun1( void (* f2)(Test3*)) { f2(this) ;} //將對象信息傳給Helper函數。 void Memberfun2( ) {printf("%s /n","Calling Test3::Memberfun2 OK"); } //成員函數2。 void Memberfun3( ) { Memberfun1( Helper);}  //… }; 

這種間接方式對成員函數沒有任何限制,克服了第一種方法成員函數不能使用任何類的成員的缺點,但由于有static 成員,類的繼承仍受到制約。

(3)使用一個全程函數(global function)為中轉間接調用類的成員函數,仍以例3為例,將代碼作如下修改(VC++6.0編譯通過):

class Test3; void __cdecl Helper(Test3* test3); class Test3 {  public: //… void Memberfun1( void (* f2)(Test3*)) { f2(this) ;} //成員函數1調用成員函數//2。 void Memberfun2( ) {printf("%s /n","Calling Test3::Memberfun2 OK"); } //成員函數2。 void Memberfun3( ) { Memberfun1( Helper);}  //… };  void __cdecl Helper(Test3* test3) {  test3->Memberfun2(); }; 

這個方法對成員函數沒有任何要求,但是需要較多的代碼。

除上述三種方法外還有其他方法,如, 可以在匯編層面上修改代碼解決上述問題等,不屬于本文范圍。

結論:函數指針不能直接調用類的成員函數,需采取間接的方法,原因是成員函數指針與一般函數指針有根本的不同,成員函數指針除包含地址信息外,同時攜帶其所屬對象信息。本文提供三種辦法用于間接調用成員函數。這三種辦法各有優缺點,適用于不同的場合。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
欧美高清视频一区二区| 欧美黑人又粗大| 69久久夜色精品国产69乱青草| 日韩欧美在线国产| 欧美一级bbbbb性bbbb喷潮片| 久久久av免费| 高跟丝袜一区二区三区| 日韩电影大全免费观看2023年上| 亚洲自拍av在线| 欧美激情第三页| 亚洲系列中文字幕| 欧美大学生性色视频| 97人人模人人爽人人喊中文字| 欧美极品美女视频网站在线观看免费| 国产精品久久久久久久电影| 亚洲精品第一页| 精品久久久999| 欧美国产日韩一区二区在线观看| 在线视频欧美日韩精品| 日韩电视剧在线观看免费网站| 精品高清美女精品国产区| 亚洲欧美在线免费| 亚洲男人天堂2024| 日本19禁啪啪免费观看www| 97精品在线视频| 午夜精品美女自拍福到在线| 色与欲影视天天看综合网| 91精品视频免费| 日韩精品视频免费专区在线播放| 国产精品成人一区二区三区吃奶| 国产精品久久久久一区二区| 九九热精品视频国产| 91av在线网站| 人人爽久久涩噜噜噜网站| 亚洲欧洲在线视频| 黄色成人在线免费| 欧美成人激情视频免费观看| 亚洲乱亚洲乱妇无码| 日本欧美黄网站| 蜜臀久久99精品久久久无需会员| 精品久久久在线观看| 国产欧美欧洲在线观看| 色噜噜国产精品视频一区二区| 欧美成人三级视频网站| 国模精品一区二区三区色天香| 日韩美女免费观看| 亚洲自拍偷拍第一页| 91久久精品视频| 欧美高清视频免费观看| 91精品在线观看视频| 日韩免费观看视频| 久久深夜福利免费观看| 精品亚洲精品福利线在观看| 国产精品成av人在线视午夜片| 亚洲自拍小视频| 不卡av电影院| 精品国产欧美一区二区五十路| 91精品国产精品| 国产精品免费视频久久久| 黑人巨大精品欧美一区二区一视频| 91手机视频在线观看| 另类少妇人与禽zozz0性伦| 亚洲欧美制服另类日韩| 在线日韩欧美视频| 久久香蕉精品香蕉| 欧美电影在线观看高清| 91精品国产综合久久香蕉最新版| 91精品国产91| 日本精品中文字幕| 亚洲人成77777在线观看网| 在线性视频日韩欧美| 97**国产露脸精品国产| 久久精品成人一区二区三区| 亚洲一区二区三区久久| 欧美噜噜久久久xxx| 亚洲高清免费观看高清完整版| 亚洲字幕一区二区| 欧美放荡办公室videos4k| 国产精品影院在线观看| 欧美大片欧美激情性色a∨久久| 成人性教育视频在线观看| 一区二区国产精品视频| 91精品久久久久久久久不口人| 欧美成人精品在线播放| 91嫩草在线视频| 日韩视频―中文字幕| 亚洲天堂av女优| 国产激情久久久| 色婷婷av一区二区三区在线观看| 日本韩国欧美精品大片卡二| 亚洲www永久成人夜色| 8x海外华人永久免费日韩内陆视频| 久久精品一偷一偷国产| 日本高清视频一区| 欧美日韩一区二区三区| 日韩美女写真福利在线观看| 国产精品香蕉av| 一区二区在线视频| 中文字幕亚洲综合久久筱田步美| 2019国产精品自在线拍国产不卡| 国产亚洲精品美女| 亚洲性猛交xxxxwww| 疯狂蹂躏欧美一区二区精品| 久久资源免费视频| 欧美怡春院一区二区三区| 91成人在线观看国产| 国产精品一区二区在线| 国产精品久久久久秋霞鲁丝| 欧美日韩国产成人在线观看| 国内免费久久久久久久久久久| 日韩av在线免费观看一区| 久久中国妇女中文字幕| 日韩高清电影免费观看完整| 成人精品福利视频| 日本sm极度另类视频| 日韩一区在线视频| 国产欧美精品日韩精品| 欧美日韩激情网| 欧美人在线观看| 国产欧美久久一区二区| 黄网动漫久久久| 久久精品夜夜夜夜夜久久| 亚洲最大成人网色| 久久伊人91精品综合网站| 91av在线播放| 欧美精品www在线观看| 亚洲韩国青草视频| 色偷偷88888欧美精品久久久| 欧美性理论片在线观看片免费| xvideos亚洲| 欧美午夜片欧美片在线观看| 日韩av日韩在线观看| 精品国产乱码久久久久酒店| 亚洲精品视频久久| 在线视频日本亚洲性| 欧美亚洲国产成人精品| 午夜精品蜜臀一区二区三区免费| 久久精品成人欧美大片| 欧美最顶级丰满的aⅴ艳星| 在线日韩精品视频| 亚洲欧美中文字幕在线一区| 国产一区二区三区高清在线观看| 日本免费在线精品| 午夜欧美大片免费观看| 国产视频在线观看一区二区| 国产精品毛片a∨一区二区三区|国| 久久亚洲综合国产精品99麻豆精品福利| 亚洲美腿欧美激情另类| 一区二区欧美激情| 91精品国产网站| 成人精品一区二区三区电影免费| 欧美在线观看网址综合| 久久综合国产精品台湾中文娱乐网| 欧美性极品xxxx娇小| 中文字幕亚洲一区二区三区| 欧美黄色成人网| 成人中心免费视频| 欧美成人激情视频免费观看| 国产成人精品免高潮费视频| 一区二区三区久久精品| 日韩成人高清在线| 中文字幕综合在线| 精品欧美aⅴ在线网站| 国产精品久久99久久|