• <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 閱讀(1090) 評論(11)  編輯 收藏 引用 所屬分類: 自家破爛C++
            九九精品99久久久香蕉| 狠狠综合久久综合中文88| 久久香蕉超碰97国产精品| 2020久久精品国产免费| 国产亚洲色婷婷久久99精品91| 国产99久久久国产精品小说| 日韩av无码久久精品免费| 精品国产婷婷久久久| 婷婷综合久久中文字幕蜜桃三电影| 国产高清国内精品福利99久久| 青青草原精品99久久精品66| 日韩精品无码久久一区二区三| 亚洲国产精品无码久久久秋霞2| 久久WWW免费人成—看片| 国产精品久久久久影院色| 欧美国产成人久久精品| 久久久久无码国产精品不卡| 欧美激情精品久久久久| 人妻无码αv中文字幕久久琪琪布| 久久精品一区二区影院| 国产91久久综合| 久久综合久久综合久久| 国产人久久人人人人爽| 久久精品www人人爽人人| 久久久久久精品免费免费自慰| 日本加勒比久久精品| 色综合久久天天综线观看| 色8激情欧美成人久久综合电| 久久99亚洲综合精品首页| 狠狠人妻久久久久久综合蜜桃| 国产精品欧美久久久久天天影视| 色综合久久精品中文字幕首页| 麻豆亚洲AV永久无码精品久久| 久久亚洲中文字幕精品有坂深雪| 亚洲AV日韩AV永久无码久久| 亚洲国产精品成人久久| 精品熟女少妇a∨免费久久| 久久精品国产亚洲麻豆| 精品久久一区二区三区| 久久精品国产精品亜洲毛片| 性高湖久久久久久久久AAAAA|