• <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>

            sanxcoo

            做一個(gè)耐得住寂寞的人
            posts - 1, comments - 8, trackbacks - 0, articles - 0

            在網(wǎng)上查了一些資料,做了一個(gè)Thunk模板,能夠正確調(diào)用成員函數(shù)。但是在做取成員函數(shù)地址操作時(shí)比較麻煩,需要用到匯編。
            //取成員函數(shù)地址
            DWORD_PTR off = 0;
            _asm
            {
               mov eax, Class::MemFunc
               mov DWORD PTR [off], eax
            }
            每指定一個(gè)成員函數(shù)作為一個(gè)回調(diào)函數(shù)就要做如上操作。本想將Class和MemFunc作為兩個(gè)參數(shù)定義一個(gè)宏包含匯編語(yǔ)言部分,
            但不知道怎樣編寫匯編部分。如果有知道怎么編寫的,希望能不吝賜教^_^

            // Thunk 具體實(shí)現(xiàn)
            #pragma pack( push, 1 )
            struct Thunk_struct
            {
             BYTE  op_movecx;  // as operation "mov" in asm
             DWORD_PTR val_ecx;
             BYTE  op_call;  // as operation "jmp" in asm
             DWORD_PTR val_address;
            };
            #pragma pack( pop )

            template < class TCallback, class TClass >
            class Thunk
            {
            public:
             TCallback MemFuncToCallback( TClass* pObject, DWORD_PTR pMemFuncAddress )
             {
              // 0xB9是“mov ecx, 數(shù)值”的機(jī)器碼
              m_thunk.op_movecx = 0xB9;
              // 將對(duì)象指針pObject賦值給ecx
              m_thunk.val_ecx = (DWORD_PTR)pObject;
              // 0xE9是“jmp 相對(duì)地址”的機(jī)器碼
              m_thunk.op_call = 0xE9;
              // 利用成員函數(shù)的具體地址pMemFuncAddress計(jì)算jmp的相對(duì)地址
              m_thunk.val_address = pMemFuncAddress - ((DWORD_PTR)(&m_thunk.val_address) + sizeof(DWORD_PTR));

              return (TCallback)&m_thunk;
             };

            protected:
             Thunk_struct m_thunk;
            };

            Feedback

            # re: 將成員函數(shù)作為回調(diào)函數(shù)  回復(fù)  更多評(píng)論   

            2010-04-01 17:36 by OwnWaterloo
            取成員指針地址:

            template<typename D,typename S>
            D cast_union(S src)
            {
            union
            {
            S src;
            D dst;
            } u = {src};
            return u.dst;
            }

            uintptr_t address = cast_union<uintptr_t>(&C::F);

            # re: 將成員函數(shù)作為回調(diào)函數(shù)  回復(fù)  更多評(píng)論   

            2010-04-01 17:41 by OwnWaterloo
            注意, 成員指針并不一定僅僅是地址。
            所以最安全的作法是實(shí)現(xiàn)一個(gè)轉(zhuǎn)發(fā)函數(shù), 而不是去取地址:

            void forwarding(void* o, int i, double d)
            {
            static_cast<C*>(o)->F(i, d);

            }

            然后將forwarding綁定到一個(gè)object上:
            C o;
            void (*f)(int, double ) = bind(forwarding, &o);

            然后將f傳遞給需要回調(diào)的地方:
            f(1212, 3.26);

            等效于:
            o.f(1212, 326);

            # re: 將成員函數(shù)作為回調(diào)函數(shù)  回復(fù)  更多評(píng)論   

            2010-04-01 18:26 by Sanxcoo
            @OwnWaterloo
            謝謝。
            成員指針如果不是地址的話,還有其他哪些情況?希望能舉個(gè)例子

            # re: 將成員函數(shù)作為回調(diào)函數(shù)  回復(fù)  更多評(píng)論   

            2010-04-01 18:31 by OwnWaterloo
            @Sanxcoo
            gcc下有繼承關(guān)系。
            msvc下有多繼承關(guān)系。

            指向成員的指針就可能不僅僅是一個(gè)地址。

            # re: 將成員函數(shù)作為回調(diào)函數(shù)  回復(fù)  更多評(píng)論   

            2010-04-02 11:24 by 陳梓瀚(vczh)
            使用std::function完美解決這個(gè)問(wèn)題

            # re: 將成員函數(shù)作為回調(diào)函數(shù)  回復(fù)  更多評(píng)論   

            2010-04-02 15:56 by OwnWaterloo
            @陳梓瀚(vczh)
            std::function?
            那是什么東西?

            # re: 將成員函數(shù)作為回調(diào)函數(shù)  回復(fù)  更多評(píng)論   

            2010-04-02 22:56 by Sanxcoo
            應(yīng)該是std::tr1::function

            # re: 將成員函數(shù)作為回調(diào)函數(shù)  回復(fù)  更多評(píng)論   

            2010-04-03 19:49 by OwnWaterloo
            @Sanxcoo
            哈, 既然你已經(jīng)找了這么復(fù)雜的方案, 相信你是清楚tr1::function這種東西是無(wú)效的~_~

            tr1::function對(duì)象或者std::binder1st, std::bind2nder對(duì)象的接受者都是template。
            而當(dāng)接收者只能接受pointer to function時(shí), tr1::function, binder這種東西都是無(wú)效的~


            這里有一個(gè)類似的, 不過(guò)不需要寫匯編。
            http://code.google.com/p/callback-currying/source/browse/trunk/src/stdcall2stdcall.c

            項(xiàng)目新的地址在
            http://github.com/OwnWaterloo/callback-currying

            不過(guò)目前是空的……

            只有注冊(cè)用戶登錄后才能發(fā)表評(píng)論。
            網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問(wèn)   Chat2DB   管理


            新狼窝色AV性久久久久久| 久久精品国产亚洲av瑜伽| 久久久国产99久久国产一| 久久精品中文无码资源站| 久久综合香蕉国产蜜臀AV| 国产亚洲美女精品久久久| 国产精品久久久久久久app| 久久99国产精品尤物| 青青青伊人色综合久久| 合区精品久久久中文字幕一区| 无码人妻精品一区二区三区久久| 久久国产精品成人免费| 久久婷婷是五月综合色狠狠| 成人综合伊人五月婷久久| 久久免费视频6| 久久精品国产91久久麻豆自制| 武侠古典久久婷婷狼人伊人| 久久国产精品久久久| 欧美日韩精品久久久免费观看| 国产亚洲精久久久久久无码AV| 久久一日本道色综合久久| 人妻无码精品久久亚瑟影视| 99热成人精品热久久669| 狠狠综合久久AV一区二区三区| 91秦先生久久久久久久| 77777亚洲午夜久久多人| 久久精品亚洲男人的天堂| 国产精品18久久久久久vr| 亚洲精品乱码久久久久久蜜桃不卡 | 色综合久久久久网| 久久国产精品无码HDAV| 97精品依人久久久大香线蕉97| 武侠古典久久婷婷狼人伊人| 久久久久久久久久免免费精品| 狠狠久久综合| 久久精品视屏| 久久免费视频6| 久久午夜夜伦鲁鲁片免费无码影视| 色8激情欧美成人久久综合电| 久久99亚洲综合精品首页| 久久久99精品成人片中文字幕|