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

            qiezi的學習園地

            AS/C/C++/D/Java/JS/Python/Ruby

              C++博客 :: 首頁 :: 新隨筆 ::  ::  :: 管理 ::
            很久以前看到有人問“如何在C++中實現動態加載類”時,簡單地做了一個。

            不過當時沒有去考慮動態加載DLL的情況。

            今天在cpp@codingnow.com中也有人問到這個問題,就把它給做完了。

            當然只是簡單地做到了“從全局類型庫中,根據類名創建實例,支持動態DLL加載”,說得更明白點:

            在應用程序App1中,向全局類型庫中注冊一個類型"Test",在另一個隱式鏈接的DLL中(即App1一啟動就加載的DLL),向全局類型庫中注冊另外幾個類型。這時可以在App1中通過類型的名字生成實例。

            在另一個顯式加載的DLL中(即調用LoadLibrary加載),向全局類型庫中注冊其它幾個類型。這時通過LoadLibrary加載這個DLL,就可以生成這幾個類型的實例了。

            這地方不能上傳文件,就把代碼貼一點吧:

            typelib.h文件:

            #ifndef __TYPE_LIB_H__
            #define __TYPE_LIB_H__

            typedef 
            void*(*CREATE_FUNC)();
            typedef 
            void(*RELEASE_FUNC)(void*);

            void regtype (const char* name, CREATE_FUNC cfunc, RELEASE_FUNC rfunc);

            void* createObject (const char* name);

            void releaseObject (const char* name, void* p);

            struct ITestInterface
            {
                
            virtual ~ITestInterface () {}
                
            virtual void print () const = 0;
            };

            template 
            <class T>
            void* create ()
            {
                
            return new T;
            }

            template 
            <class T>
            void release (void* p)
            {
                delete (T
            *)p;
            }

            #endif // __TYPE_LIB_H__

            typelib.cpp文件:

            #include "typelib.h"

            #include 
            <string>
            #include 
            <map>
            using namespace std;

            namespace TypeRegistry
            {
                
            static map < string, pair<CREATE_FUNC, RELEASE_FUNC> >  types_info;

                template 
            <class T>
                
            void regType (const string& name)
                {
                    types_info.insert (make_pair(name, make_pair(create
            <T>, release<T>)));
                }
            }

            void regtype (const char* name, CREATE_FUNC cfunc, RELEASE_FUNC rfunc)
            {
                TypeRegistry::types_info.insert (make_pair(name, make_pair(cfunc, rfunc)));
            }

            void* createObject (const char* name)
            {
                map 
            < string, pair<CREATE_FUNC, RELEASE_FUNC> >::const_iterator iter;
                iter 
            = TypeRegistry::types_info.find (name);
                
            if (iter != TypeRegistry::types_info.end ())
                    
            return (*iter->second.first)();
                
            return NULL;
            }

            void releaseObject (const char* name, void* p)
            {
                map 
            < string, pair<CREATE_FUNC, RELEASE_FUNC> >::const_iterator iter;
                iter 
            = TypeRegistry::types_info.find (name);
                
            if (iter != TypeRegistry::types_info.end ())
                    (
            *iter->second.second)(p);
            }

            把它編譯成靜態lib或DLL,就可以使用了。

            在那2個為我們提供類型的DLL中,DllMain函數中加入下面的代碼:

            // FirstTest和SecondTest是2個類名
            regtype("FirstTest", create<FirstTest>, release<FirstTest>);
            regtype(
            "FirstTest", create<SecondTest>, release<SecondTest>);

            就可以向全局類型庫中注冊類型。注意在類型庫中是沒有保存類信息的,所以最好是使用單根類庫來做。

            下面是一點測試代碼:

            int main()
            {
                   
            // 程序啟動時注冊類型。
                   
            // 實際上啟動時就加載了另一個動態鏈接庫,那里面有3個類型,所以現在有4個類型
                   regtype ("MyTest", create<MyTest>, release<MyTest>);
                   
            while (1)
                   {
                           
            string class_name;
                           cin 
            >> class_name;
                           
            if (class_name == "q")
                                   
            break;
                           
            // 當輸入load時,把另一個動態鏈接庫加載進來,那個鏈接庫中有2個類型,現在共有6個類型可用。
                           if (class_name == "load")
                           {
                                   LoadLibrary(
            "typelibdll_test.dll");
                                   
            continue;
                           }
                           ITestInterface
            * test = (ITestInterface*)createObject (class_name.c_str());
                           
            if (!test)
                           {
                                   cout 
            << "This type not found" << endl;
                                   
            continue;
                           }
                           test
            ->print ();
                           releaseObject (class_name.c_str(), test);
                   }
                   
            return 0;
            }

            還有一個沒考慮的地方,就是沒有給它加鎖,因為有可能在一個線程中加載一個DLL。

            不過我還有些懷疑這東西是否真的有用?
            posted on 2005-09-26 17:31 qiezi 閱讀(1083) 評論(11)  編輯 收藏 引用 所屬分類: 自家破爛C++
            AV无码久久久久不卡蜜桃| 97久久国产亚洲精品超碰热| 久久国产成人亚洲精品影院| 欧美与黑人午夜性猛交久久久 | 青青草国产精品久久久久| 国产精品无码久久四虎| 久久久久久伊人高潮影院| 久久国产乱子伦精品免费强| 日本久久久久久久久久| 国产婷婷成人久久Av免费高清| 久久精品国产只有精品66| 精品久久久久中文字幕日本| 久久久久18| 久久精品国产99国产精品澳门| 一本大道久久香蕉成人网| 久久精品一区二区国产| 久久精品中文无码资源站| 久久国产精品国语对白| 国产一久久香蕉国产线看观看| 久久久久av无码免费网| 无码国内精品久久人妻麻豆按摩| 99精品久久精品| 亚洲色欲久久久综合网| 久久大香萑太香蕉av| 丰满少妇人妻久久久久久4| 人妻精品久久无码专区精东影业| 色8激情欧美成人久久综合电| 国产精品久久久久久久午夜片| 久久国产精品99精品国产| 色欲综合久久躁天天躁蜜桃| 精品伊人久久久| 久久无码高潮喷水| 久久精品国产清自在天天线 | 99re久久精品国产首页2020| 99精品国产99久久久久久97| 久久久久久久精品妇女99| 久久综合久久美利坚合众国| 香蕉久久久久久狠狠色| 亚洲国产成人精品女人久久久 | 人妻无码精品久久亚瑟影视| 色8激情欧美成人久久综合电|