• <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++中實現(xiàn)動態(tài)加載類”時,簡單地做了一個。

            不過當時沒有去考慮動態(tài)加載DLL的情況。

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

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

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

            在另一個顯式加載的DLL中(即調(diào)用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);
            }

            把它編譯成靜態(tài)lib或DLL,就可以使用了。

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

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

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

            下面是一點測試代碼:

            int main()
            {
                   
            // 程序啟動時注冊類型。
                   
            // 實際上啟動時就加載了另一個動態(tài)鏈接庫,那里面有3個類型,所以現(xiàn)在有4個類型
                   regtype ("MyTest", create<MyTest>, release<MyTest>);
                   
            while (1)
                   {
                           
            string class_name;
                           cin 
            >> class_name;
                           
            if (class_name == "q")
                                   
            break;
                           
            // 當輸入load時,把另一個動態(tài)鏈接庫加載進來,那個鏈接庫中有2個類型,現(xiàn)在共有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 閱讀(1077) 評論(11)  編輯 收藏 引用 所屬分類: 自家破爛 、C++
            久久精品二区| 精品久久久久久国产| 香蕉久久永久视频| 精品久久久久久无码中文野结衣 | 国产精品99久久久精品无码 | 青青青国产成人久久111网站| 国产精品内射久久久久欢欢| 亚洲国产天堂久久综合| 精品久久久久久国产潘金莲| 国产综合免费精品久久久| 久久精品国产免费观看三人同眠| 国产情侣久久久久aⅴ免费| 欧美国产精品久久高清| 韩国无遮挡三级久久| 久久天天躁夜夜躁狠狠| 久久国产美女免费观看精品 | 久久精品亚洲精品国产色婷| 国产综合成人久久大片91| 久久99精品久久久久久动态图| 日韩亚洲国产综合久久久| 国产91久久精品一区二区| 无码专区久久综合久中文字幕| 亚洲午夜福利精品久久| 狠狠久久综合伊人不卡| 99久久国产免费福利| 91精品国产高清久久久久久io | 久久99热只有频精品8| 久久精品成人欧美大片| 免费久久人人爽人人爽av| 日日狠狠久久偷偷色综合96蜜桃| 国产免费久久精品99久久| 久久成人国产精品二三区| 国产亚洲婷婷香蕉久久精品| 亚洲AV无码成人网站久久精品大| 亚洲精品无码久久久| 一本一道久久a久久精品综合| 久久久久无码专区亚洲av| 久久亚洲精品无码播放| 亚洲欧美日韩久久精品| 久久久亚洲精品蜜桃臀| 怡红院日本一道日本久久 |