• <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++
            无码国内精品久久人妻| 观看 国产综合久久久久鬼色 欧美 亚洲 一区二区 | 91亚洲国产成人久久精品| 久久r热这里有精品视频| 久久精品国产亚洲一区二区三区| 久久久久综合中文字幕| 一级a性色生活片久久无少妇一级婬片免费放 | 久久精品麻豆日日躁夜夜躁| 69SEX久久精品国产麻豆| 久久久无码精品亚洲日韩软件| 久久青青草原精品国产| 99久久精品免费看国产一区二区三区 | 国内精品伊人久久久久av一坑| 亚洲狠狠综合久久| 亚洲国产精品无码久久久蜜芽 | 亚洲国产高清精品线久久| 99久久久精品免费观看国产| 国内精品久久久久久中文字幕| 99久久成人国产精品免费| 久久91精品国产91| 国内精品伊人久久久久妇| 91精品国产色综久久| 久久亚洲私人国产精品| 欧美国产精品久久高清| 中文字幕久久欲求不满| 久久av无码专区亚洲av桃花岛| 日本WV一本一道久久香蕉| 蜜臀久久99精品久久久久久| 精品999久久久久久中文字幕| 无码人妻精品一区二区三区久久久| 久久亚洲国产精品123区| 人人狠狠综合久久亚洲婷婷| 久久久国产乱子伦精品作者| 久久精品中文字幕一区| 国产精品久久久香蕉| 亚洲AV伊人久久青青草原| 亚洲国产成人久久综合碰碰动漫3d| 国产精品日韩深夜福利久久 | 亚洲欧美精品一区久久中文字幕| 亚洲狠狠久久综合一区77777| 99精品伊人久久久大香线蕉|