青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

山寨:不是最好的,是最適合我們的!歡迎體驗山寨 中文版MSDN

Blog @ Blog

當華美的葉片落盡,生命的脈絡(luò)才歷歷可見。 -- 聶魯達

常用鏈接

統(tǒng)計

積分與排名

BBS

Blog

Web

最新評論

[轉(zhuǎn)]C++指針直接調(diào)用類成員函數(shù)探討

在編程工作中常會遇到在一個“類”中通過函數(shù)指針調(diào)用成員函數(shù)的要求,如,當在一個類中使用了C++標準庫中的排序函數(shù)qsort時,因qsort參數(shù)需要一個“比較函數(shù)”指針,如果這個“類”使用某個成員函數(shù)作“比較函數(shù)”,就需要將這個成員函數(shù)的指針傳給qsort供其調(diào)用。本文所討論的用指針調(diào)用 “類”的成員函數(shù)包括以下三種情況:

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

#include <stdlib.h>

typedef 
void (*Function1)( ); //定義一個函數(shù)指針類型。
Function1 f1;

class Test1
{
 
public
  
//…被調(diào)用的成員函數(shù)。
  void Memberfun1( ){ printf("%s \n","Calling Test3::Memberfun2 OK");}// 
  void Memberfun2()
  
{
   f1
=reinterpret_cast<Function1>(Memberfun1);//將成員函數(shù)指針賦予f1。編譯出錯。
   f1();
  }

  
//
}
;

int main()
{
 Test1 t1;
 t1.Memberfun2();
 
return 0;
}
 


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

#include <stdlib.h>

class Test2
{
private: 
int data
[2]; 
//…
public:
//…
int __cdecl Compare(const void* elem1
, const void* elem2) //成員函數(shù)。

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); //標準庫函數(shù)調(diào)用成
//員函數(shù)。編譯出錯。


//…
}
;

int main( )
{
Test2 t2
;
t2.Memberfun(); //調(diào)用成員函數(shù)。
return 0;



(3)同一個“類”內(nèi),一個成員函數(shù)調(diào)用另一個成員函數(shù), 如:

#include "stdlib.h"
class Test3
{
public:
//
void Memberfun1( void (* f2)( ) ) { f2( ) ;} //成員函數(shù)1調(diào)用成員函數(shù)//2。
void Memberfun2( ) { printf("%s \n","Calling Test3::Memberfun2 OK");} //成員函數(shù)2。
void Memberfun3( ) { Memberfun1( Memberfun2);} // 編譯出錯 
//
}
;

int main( )


Test3 t3;
t3.Memberfun3(); 
//調(diào)用成員函數(shù)。
return 0;

}
 

  以上三種情況的代碼語法上沒有顯著的錯誤,在一些較早的編譯環(huán)境中,如,VC++ 4.0, 通常可以編譯通過,或至多給出問題提醒(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參數(shù)中所調(diào)用的函數(shù)類型不對。

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

#include <stdlib.h>

void Memberfun1( void (* f2)( ) ) { f2( ) ;} //原成員函數(shù)1調(diào)用成員函數(shù)//2。
void Memberfun2( ) { printf("%s \n","Calling Test3::Memberfun2 OK");} //原成員函數(shù)2。
void Memberfun3( ) { Memberfun1( Memberfun2);} 

int main( )
{
Memberfun3 ();
return 0;
}
 


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

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

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

#include "stdafx.h"
#include <iostream>
#include 
<typeinfo.h>

class Test; //一個未定義的類。

class Test2 //一個空類。
{
}
;

class Test3 //一個有定義的類。
{
 
public:
  
//
  void (* memberfun)();
  
void Memberfun1( void (* f2)( ) ) { f2( ) ;} //成員函數(shù)1調(diào)用成員函數(shù)//2。
  void Memberfun2( );//成員函數(shù)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 
<<"一般函數(shù)指針長度= "<< sizeof(void(*)()) << '\n';
 std::cout 
<<"-類的成員函數(shù)指針長度-"<<'\n'<<'\n';
 std::cout 
<<"Test3類成員函數(shù)指針長度="<< sizeof(void(Test3::*)())<<'\n'<<'\n';
 std::cout 
<<"Test5類成員函數(shù)指針長度="<<sizeof(void (Test5:: *)())<<'\n';
 std::cout 
<<"Test4類成員函數(shù)指針長度="<<sizeof(void (Test4:: *)())<<'\n';
 std::cout 
<<"Test類成員函數(shù)指針長度="<<sizeof(void(Test::*)()) <<'\n';
 
return 0;
}
 

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

  一般非成員函數(shù)指針長度= 4
  
  -類的成員函數(shù)指針長度-

  Test3類成員函數(shù)指針長度=4
  Test5類成員函數(shù)指針長度=8
  Test4類成員函數(shù)指針長度=12
  Test類成員函數(shù)指針長度=16

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

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

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

  (1) 將需要調(diào)用的成員函數(shù)設(shè)為static 類型,如:在前述例子2中,將class Test2 成員函數(shù)Compare 定義前加上static 如下(黑體為改變之處):

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

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

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

class Test3
{
 
public:
  
//
  void static __cdecl Helper(Test3* test3)
  
{
   test3
->Memberfun2();
  }

  
void Memberfun1( void (* f2)(Test3*)) { f2(this) ;} //將對象信息傳給Helper函數(shù)。
  void Memberfun2( ) {printf("%s \n","Calling Test3::Memberfun2 OK"); } //成員函數(shù)2。
  void Memberfun3( ) { Memberfun1( Helper);} 
  
//
}


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

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

class Test3;
void __cdecl Helper(Test3* test3);

class Test3
{
 
public:
  
//
  void Memberfun1( void (* f2)(Test3*)) { f2(this) ;} //成員函數(shù)1調(diào)用成員函數(shù)//2。
  void Memberfun2( ) {printf("%s \n","Calling Test3::Memberfun2 OK"); } //成員函數(shù)2。
  void Memberfun3( ) { Memberfun1( Helper);} 
  
//
}
;

void __cdecl Helper(Test3* test3)
{
 test3
->Memberfun2();
}


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

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

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

  參考文獻: [1] Bjarne Stroustrup The C++ programming language 2nd edition [M]

posted on 2007-08-09 21:08 isabc 閱讀(405) 評論(0)  編輯 收藏 引用 所屬分類: C++基礎(chǔ)

廣告信息(免費廣告聯(lián)系)

中文版MSDN:
歡迎體驗

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            亚洲图片欧美日产| 一区二区三区国产在线| 亚洲欧美日韩国产一区二区三区 | 亚洲午夜未删减在线观看| 欧美精品九九| 欧美日韩在线第一页| 国产精品视频免费| 国产亚洲精品高潮| 亚洲精华国产欧美| 99热这里只有精品8| 亚洲欧美一区二区三区极速播放 | 久久伊人亚洲| 在线精品视频一区二区三四| 久久亚洲二区| 久久精品国产亚洲一区二区三区| 91久久综合亚洲鲁鲁五月天| 久久精品视频免费| 亚洲无玛一区| 欧美激情va永久在线播放| 91久久精品国产91久久| 欧美黄色大片网站| 欧美日本中文| 久久国产精品免费一区| 欧美高清一区| 国产真实久久| 午夜精品久久久99热福利| 欧美一区二区三区四区在线| 欧美人成在线| 久久精品99国产精品| 欧美刺激午夜性久久久久久久| 国产综合精品| 日韩视频精品在线观看| 老鸭窝91久久精品色噜噜导演| 亚洲视频香蕉人妖| 亚洲国产一区二区a毛片| 久久久久综合网| 一区二区在线观看视频| 久久久美女艺术照精彩视频福利播放 | 久久精品国语| 欧美精品二区| 你懂的视频一区二区| 午夜视频一区二区| 国产精品一区二区黑丝| 亚洲免费一在线| 欧美国产精品人人做人人爱| 久久精品欧洲| 国产精品久久久久免费a∨| 一区二区三区毛片| 一区二区三区欧美激情| 亚洲激情综合| 久久久在线视频| 久久精品国产99国产精品| 欧美日韩一二三区| 亚洲黄色片网站| 最新日韩在线视频| 亚洲国产欧美日韩精品| 在线看片欧美| 久久久久久有精品国产| 国产精品久久一级| 欧美一级视频| 亚洲在线免费| 黄色成人精品网站| 欧美电影免费观看高清| 欧美高清视频免费观看| 免费不卡亚洲欧美| 欧美伦理91| 亚洲人久久久| 国产区精品视频| 亚洲电影成人| 国产精品任我爽爆在线播放 | 久久精品一区二区三区中文字幕| 久久婷婷激情| 免费亚洲一区| 亚洲免费av片| 欧美一区二区三区另类| 久久xxxx| 欧美成人免费全部观看天天性色| 欧美成人第一页| 亚洲精品视频免费观看| 午夜精品久久久久久久男人的天堂| 在线观看视频一区| 国产精品99久久久久久www| 在线播放一区| 欧美激情第五页| 亚洲最快最全在线视频| 狠狠色狠色综合曰曰| 久久久久久一区| 亚洲国产精品一区二区第四页av| 99精品国产高清一区二区 | 欧美国产日本| 宅男噜噜噜66一区二区66| 国产精品视频免费一区| 羞羞色国产精品| 欧美成年人网站| 亚洲一区在线免费观看| 欧美 日韩 国产在线| 久久亚洲精品一区| 亚洲精品永久免费| 国产欧美另类| 欧美激情麻豆| 久久gogo国模裸体人体| 久久精品国产77777蜜臀| 亚洲激情在线播放| 国产精品网曝门| 老巨人导航500精品| 中文一区字幕| 欧美激情在线| 久久久av水蜜桃| 亚洲午夜在线| 亚洲激情视频网| 激情五月综合色婷婷一区二区| 欧美日韩国产不卡| 久久色在线播放| 亚洲一级网站| 日韩视频精品在线观看| 免费久久99精品国产| 午夜精品亚洲一区二区三区嫩草| 亚洲国产福利在线| 欧美成人激情视频| 欧美一级视频| 亚洲深夜福利在线| 亚洲精品欧美| 亚洲男人的天堂在线aⅴ视频| 亚洲国产福利在线| 黄色成人在线观看| 亚洲视频香蕉人妖| 欧美激情视频在线免费观看 欧美视频免费一| 亚洲性xxxx| 亚洲免费观看在线观看| 欧美v国产在线一区二区三区| 久久爱www久久做| 亚洲网站啪啪| 在线一区二区三区四区五区| 亚洲理伦在线| 亚洲六月丁香色婷婷综合久久| 一区二区三区在线观看欧美| 国产欧美日韩精品丝袜高跟鞋| 国产精品久久亚洲7777| 欧美视频一区二区在线观看 | 一区二区免费在线视频| 亚洲人永久免费| 亚洲国产专区| 91久久嫩草影院一区二区| 欧美激情无毛| 亚洲国产一区视频| 日韩视频中文字幕| 夜夜嗨av色一区二区不卡| 亚洲精品在线免费| 亚洲毛片在线看| 亚洲图片在线| 欧美一区二区三区久久精品茉莉花| 亚洲欧美在线一区二区| 欧美一区二区在线免费播放| 欧美一区成人| 免费成人黄色| 性欧美xxxx视频在线观看| 亚洲欧美一级二级三级| 欧美一站二站| 欧美va亚洲va国产综合| 欧美精品一区二区三区很污很色的| 欧美片在线观看| 国产欧美丝祙| 亚洲国产精品一区二区第一页| 亚洲精品日本| 亚洲欧美在线aaa| 久久久欧美精品sm网站| 欧美激情亚洲精品| 99视频精品在线| 欧美在线亚洲综合一区| 美女脱光内衣内裤视频久久网站| 欧美精品久久天天躁| 国产精品久久久久一区二区三区 | 在线亚洲免费| 久久久久久亚洲精品中文字幕| 欧美成人在线免费观看| 一区二区三区国产在线| 久久超碰97人人做人人爱| 欧美国产1区2区| 国产日韩精品一区二区三区在线 | 欧美第一黄色网| 国产精品青草综合久久久久99 | 性刺激综合网| 欧美岛国激情| 午夜免费电影一区在线观看| 久久综合福利| 国产乱子伦一区二区三区国色天香| 亚洲国产另类久久久精品极度| 亚洲视频二区| 欧美黄色大片网站| 欧美一级久久久久久久大片| 欧美精彩视频一区二区三区| 国产一在线精品一区在线观看| 国产日韩精品一区二区浪潮av| 免费短视频成人日韩| 美女诱惑一区| 国产日韩欧美制服另类| 一本色道久久综合亚洲精品小说 | 亚洲动漫精品| 久久不见久久见免费视频1| 亚洲精品视频在线看|