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

隨筆 - 13  文章 - 36  trackbacks - 0
<2009年3月>
22232425262728
1234567
891011121314
15161718192021
22232425262728
2930311234

常用鏈接

留言簿(2)

隨筆檔案

友情鏈接

搜索

  •  

最新評論

閱讀排行榜

評論排行榜

 轉載自:http://blog.csdn.net/akirya/archive/2009/01/18/3825040.aspx


很多人對成員函數指針有無解,以為成員函數指針同普通的函數指針區別不大,多了一個參數.
然而實際上卻不是.多了個參數,是不假,但他確實不是指針,雖說名字中有指針兩個字,但實際上卻不是指針.
先看看最簡單的使用

class test
{
public:
 void func(){printf("call test::func\n");};
};

int main()
{
 void(test::*p)() = &test::func;
 test x;
 (x.*p)();
}

這里的用法是最常見的,跟普通的函數指針使用的地方也差不多。一般見到的地方也差不多都這么用。
但成員函數指針還有更好用的地方,看下面這個例子
class base
{
public:
 virtual void func(){printf("call base::func\n");};
};
class test:public base
{
public:
 void func(){printf("call test::func\n");};
};

int main()
{
 void(base::*p)() = &base::func;
 test x;
 (x.*p)();//調用的是test::func;
 base y;
 (y.*p)();//調用的是base::func;
}

從這個例子看,成員函數指針也可以使用多態。看到這里是不是覺得成員函數指針跟之前想的不一樣?

緊接著上面的例子,只更改main函數的內容

int main()
{
 void(base::*p)() = &base::func;
 printf("sizeof( void(base::*p)()) %d\n" , sizeof(p) );
};


這里的輸出結果能想到么?(我的系統是32位的XP)
VC9 下是 4
GCC4.2.1 是 8
CodeGear C++ 6.10 的結果是12
到這里是不是會想到如何得到成員函數的地址呢?
我想到最簡單的辦法就是輸出map文件,這樣就直接找到對應的函數的地址。
那運行時候怎么得到呢?
我的答案是沒有,
如果是虛函數則可以通過搜索虛表來得到,不過這個方法太不通用,也很難實際應用。算是一種理論上能,但不能實際應用的方法。
非虛函數的話,還不知道有什么好的方法能夠得到。

發表于 @ 2009年01月18日 10:19:00|評論(loading... )|收藏

新一篇: 如何寫易于使用的代碼 (一個宏引起的問題) | 舊一篇: 注冊COM到單獨用戶

評論

#taodm 發表于2009-01-19 15:12:59  IP: 10.63.110.*
C++語言當初設計的時候就認為你不需要得到它的真正地址。
#taodm 發表于2009-01-20 15:09:58  IP: 10.63.110.*
C++在設計的時候就認為不(應該)需要獲得成員函數的真實地址。
解決方案對你來說只是舉手之勞。
#akirya 發表于2009-01-20 18:43:34  IP: 58.30.180.*
呵呵~taodm 見笑了:D
最初這個問題就是因為你我才去弄明白的
#zenny_chen 發表于2009-01-20 18:58:09  IP: 218.242.128.*
akirya大俠似乎沒有解釋成員函數指針為何不是指針,難道它屬于別的什么東西?不過本人倒是覺得它應該算是指針吧。它具有指針的基本特性啊,呵呵。

另外,關于通過成員函數指針來獲得成員函數地址的方法確實比較困難。它的困難點在于必須繞過C++編譯器的類型檢查。像VC對成員函數指針類型檢查的很嚴,即使是void*類型都不能轉,也無法通過reinterpret_cast、dynamic_cast之類的來轉。
因此我這里將采取暴力手段來獲取:

#include <iostream>
using namespace std;


class Test
{
public:

void Hello(void)
{
cout << "Hello, world!" << endl;
}

void Hello(int i)
{
cout << "The answer is: " << i << endl;
}
};


template <typename T>
inline unsigned GetMemberFuncAddress(T p)
{
unsigned result;

__asm
{
mov eax, dword ptr [p]
mov dword ptr [result], eax
}

return result;
}


int main(void)
{
void (Test::* p)(void) = &Test::Hello;
void (Test::* q)(int) = &Test::Hello;

(Test().*p)();
(Test().*q)(100);

cout << "The address is: " << p << endl;
cout << "The address is: " << q << endl;

cout << "The address is: 0x" << hex << GetMemberFuncAddress(p) << endl;
cout << "The address is: 0x" << hex << GetMemberFuncAddress(q) << endl;

return 0;
}
#akirya 發表于2009-01-20 19:28:46  IP: 58.30.180.*
很遺憾,VC不代表C++

CodeGear C++ 6.10 下 sizeof(成員指針大小)結果是12
這個時候你如何輸出呢?12字節大小的指針?
#healer_kx 發表于2009-01-20 19:49:41  IP: 125.33.137.*
你們兩個真無聊,我都開始研究JS了
#zenny_chen 發表于2009-01-20 20:22:15  IP: 218.242.128.*
哈哈,healer_kx兄挺與時俱進的嘛。

呵呵,雖然沒用過CodeGear,但不管是基于怎樣的處理器都可以獲得指針值吧。只要匹配好地址類型就行。我剛才就是忘了說前提了。實際上如果要考慮到兼容性的話就將地址類型大小指明一下就行啊。
#ifdef ADDRESS_32BIT
typedef unsigned TYPE_ADDRESS;
#elif defined(ADDRESS_64BIT)
typedef unsigned long long TYPE_ADDRESS;
#elif defined(ADDRESS_96BIT)
typedef struct
{
unsigned low4;
unsigned mid4;
unsigned high4;
}TYPE_ADDRESS;
#endif

雖然我不知道CodeGear的匯編形式,不過我這里就再借助一下MASM了,呵呵:

template <typename T>
inline TYPE_ADDRESS GetMemberFuncAddress(T p)
{
TYPE_ADDRESS result;

#ifdef ADDRESS_32BIT
#define WORD_SIZE dword
#define MOV_FORM mov
#define REG_OPT eax
#elif defined(ADDRESS_64BIT)
#define WORD_SIZE mmword
#define MOV_FORM movq
#define REG_OPT mm0
#elif defined(ADDRESS_96BIT)
#define WORD_SIZE xmmword
#define MOV_FORM movdqu
#define REG_OPT xmm0
#endif

__asm
{
MOV_FORM REG_OPT, WORD_SIZE ptr [p]
MOV_FORM WORD_SIZE ptr [result], REG_OPT
}

return result;
}
#akirya 發表于2009-01-20 22:16:25  IP: 58.30.180.*
zenny_chen很專業

那假如是指針的話,那多態例子怎么解釋,對象不同調用的函數也就不用? 但成員函數函數指針的值并沒有改變。
#zenny_chen 發表于2009-01-20 23:33:34  IP: 116.234.182.*
呵呵,如果是虛函數的話這樣整肯定不行了。而且每種編譯器對于對象模型模型可能建的不一樣,而且虛函數表的建立機制也可能大相徑庭,所以訪問虛函數指針的值的方式也會不同。
我上面僅僅是針對你最后一句話的回答,呵呵。
#akirya 發表于2009-01-20 23:54:32  IP: 58.30.180.*
的確不是指針,在CodeGear C++ 6.10(也就是BCB的后續版本)存的是一個偏移量.
sizeof( 成員函數指針)大小是12也就說明了指針這種假設是不成立的,具體存的是取決與編譯器的實現。

標準也沒有對成員函數指針里面存的是啥東西做規定了,我寫這個目的就是說明一下,成員函數指針不是真正的指針,里面保存的不一定是地址。
#taodm 發表于2009-01-21 09:45:21  IP: 10.63.110.*
又有匯編高手冒出來了。接著又要探討獲取構造函數、析構函數真實地址的暴力手法了。。。。。。
#zenny_chen 發表于2009-01-21 13:31:21  IP: 218.242.128.*
哈哈,taodm真是搞笑啊。

嗯……如果是虛成員函數的話其地址是虛的嘛。所以拿到的函數地址可以認為是“虛指針”,呵呵。而且編譯器可以在編譯時斷定對象所調用的函數是否為虛函數,對象是否具有虛函數表指針。所以這些在編譯時就搞定的東西即使用暴力手段也難以獲取,呵呵。
我想說的是“虛指針”算是一種特殊的指針罷。
posted on 2009-03-18 11:06 Alex-Lee 閱讀(1078) 評論(0)  編輯 收藏 引用

只有注冊用戶登錄后才能發表評論。
網站導航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            狠狠干成人综合网| 欧美高清影院| 国产在线拍偷自揄拍精品| 欧美一区二区视频网站| 欧美一区=区| 精品91视频| 亚洲国产精品视频| 欧美激情久久久| 亚洲香蕉在线观看| 亚洲资源av| 伊人久久亚洲热| 亚洲第一天堂av| 欧美日韩国产123| 欧美一区二区三区在线免费观看| 美女爽到呻吟久久久久| 欧美xxx成人| 亚洲欧美在线一区二区| 新片速递亚洲合集欧美合集| 激情国产一区二区| 亚洲精品国产精品国自产在线| 国产精品成人观看视频国产奇米| 欧美综合激情网| 欧美国产精品v| 久久精品论坛| 欧美另类videos死尸| 亚洲免费视频成人| 麻豆成人综合网| 久久gogo国模裸体人体| 欧美高潮视频| 久久婷婷蜜乳一本欲蜜臀| 欧美老女人xx| 美女诱惑一区| 国产精品美女久久久久av超清 | 国产精品久久久久久久久久妞妞 | 欧美国产日韩a欧美在线观看| 欧美日韩一区在线| 米奇777超碰欧美日韩亚洲| 欧美视频官网| 亚洲国产婷婷香蕉久久久久久| 国产精品制服诱惑| 亚洲精品一区二区网址| 激情久久影院| 亚洲欧美日韩第一区| 一区二区三区高清| 老司机亚洲精品| 久久人人超碰| 国产欧美一区二区三区视频| 日韩网站在线观看| 亚洲高清在线观看| 久久国产日本精品| 久久国产精品一区二区三区四区 | 国产日韩精品在线| 日韩视频精品在线| 91久久精品国产91性色tv| 久久黄色级2电影| 欧美在线网站| 国产无一区二区| 亚洲一区二区三区激情| 亚洲在线视频免费观看| 欧美日韩在线观看视频| 亚洲激情亚洲| 99精品热6080yy久久| 蜜臀av性久久久久蜜臀aⅴ四虎| 久久精品在线播放| 国产综合久久| 久久精选视频| 免费欧美在线| 亚洲国产精品成人综合色在线婷婷| 久久不射网站| 久久综合一区二区| 在线观看视频亚洲| 免费观看成人网| 亚洲国产成人精品视频| 亚洲欧洲日本国产| 欧美大片一区二区三区| 亚洲黄色av| 亚洲视频精选| 国产精品亚洲一区| 欧美中文字幕不卡| 久久在线免费| 99视频精品全国免费| 国产精品高潮在线| 欧美一区亚洲一区| 欧美成人一区二区三区片免费| 最新成人av网站| 欧美网站在线观看| 欧美一区二区三区在| 亚洲成人中文| 午夜视频在线观看一区二区| 国产丝袜美腿一区二区三区| 久久久久免费| 亚洲国产小视频| 午夜久久99| 91久久精品国产91性色tv| 欧美日韩伦理在线免费| 午夜精品久久久久久久白皮肤 | 在线国产欧美| 欧美三级午夜理伦三级中文幕 | 正在播放亚洲一区| 久久在线91| 亚洲特色特黄| 樱桃视频在线观看一区| 欧美日韩一区二区免费视频| 欧美在线视频在线播放完整版免费观看| 玖玖国产精品视频| 亚洲——在线| 亚洲高清一区二区三区| 国产精品久久久久久久久久久久久久| 久久黄色影院| 一区二区三区国产| 免费观看在线综合| 午夜欧美视频| 一区二区三区成人精品| 在线观看一区欧美| 国产欧美综合一区二区三区| 欧美精品1区2区3区| 久久全国免费视频| 亚洲欧美日韩人成在线播放| 亚洲美女网站| 欧美激情麻豆| 久久综合激情| 久久久九九九九| 亚洲一级在线观看| 亚洲卡通欧美制服中文| 一区精品久久| 国产欧美三级| 国产老肥熟一区二区三区| 欧美日韩精品欧美日韩精品| 美国十次成人| 久久亚洲精品欧美| 久久成年人视频| 欧美在线免费| 欧美一区二区三区视频在线| 亚洲午夜在线视频| 99精品99久久久久久宅男| 最新中文字幕亚洲| 亚洲激情女人| 亚洲精品日韩在线观看| 欧美激情在线狂野欧美精品| 欧美成人日本| 欧美国产91| 亚洲激情网站| 亚洲精品网站在线播放gif| 亚洲成色777777女色窝| 蜜桃av久久久亚洲精品| 久久五月天婷婷| 老司机一区二区三区| 欧美成人高清视频| 亚洲大片av| 亚洲美女中出| 一本久久综合亚洲鲁鲁五月天| 日韩一级不卡| 亚洲一级黄色片| 午夜精品一区二区在线观看| 午夜精彩视频在线观看不卡| 午夜视频在线观看一区二区| 欧美一级免费视频| 久久免费黄色| 欧美老女人xx| 国产精品三区www17con| 国产原创一区二区| 伊人久久综合97精品| 亚洲欧洲精品一区二区三区波多野1战4| 91久久黄色| 亚洲女同性videos| 美女视频黄免费的久久| 亚洲国产精品电影| 99精品视频一区| 性久久久久久久久久久久| 久久精品国产清自在天天线| 噜噜噜久久亚洲精品国产品小说| 欧美黑人在线播放| 国产精品视频999| 影音先锋久久久| 一本大道久久精品懂色aⅴ| 午夜在线精品| 欧美激情一区二区三区蜜桃视频| 99这里有精品| 久久视频免费观看| 欧美日韩福利| 国外视频精品毛片| 一区二区日韩伦理片| 久久国产欧美精品| 亚洲狼人精品一区二区三区| 性欧美xxxx大乳国产app| 欧美成人嫩草网站| 国产日产欧美一区| 日韩一区二区精品视频| 久久米奇亚洲| 亚洲一区在线免费| 欧美mv日韩mv国产网站app| 国产精品影视天天线| 亚洲老司机av| 久久中文字幕一区| 亚洲网站在线| 欧美久久婷婷综合色| 亚洲大片av| 久久成年人视频| av成人毛片| 欧美精品久久久久久久久久|