• <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>
            技術著執著
            技術人士
            AFX_MANAGE_STATE(AfxGetStaticModuleState())
             

            先看一個例子:

            1、創建一個動態鏈接到MFC DLL的規則DLL,其內部包含一個對話框資源。指定該對話框ID如下:
                          #define IDD_DLL_DIALOG 2000

            2、創建一個基于對話框的mfc應用程序,它包含兩個對話框資源,IDD_UI_DIALOG和IDD_EXE_DIALOG。并將后者的ID指定如下:
                          #define IDD_EXE_DIALOG 2000
            其中前者是這個應用程序的用戶界面,單擊上面的按鈕,將彈出一個對話框。部分代碼如下:
            // in DLL
            void CDLL::ShowDlg(void)
            {
                   CDialog dlg(IDD_DLL_DIALOG); //打開ID為2000的對話框
                   dlg.DoModal();
            }
            // in EXE
            void CEXE::OnButtonClick()
            {
                   ShowDlg();
            }

            3、單擊按鈕,彈出的不是期望的DLL中的對話框IDD_DLL_DIALOG,而是應用程序中的對話框IDD_EXE_DIALOG。

            解釋:

            1、應用程序進程本身及其調用的每個DLL模塊都具有一個全局唯一的HINSTANCE句柄,它們代表了EXE或DLL模塊在進程虛擬空間中的起始地址。(進程本身的模塊句柄一般為0x400000,而DLL模塊的缺省句柄為0x10000000。如果程序同時加載了多個DLL,則每個DLL模塊都會有不同的HINSTANCE。應用程序在加載DLL時對其進行了重定位)。
            2、共享MFC DLL(或MFC擴展DLL)的規則DLL涉及到HINSTANCE句柄問題,HINSTANCE句柄對于加載資源特別重要。EXE和DLL都有其自己的資源,而且這些資源的ID可能重復,如果應用程序與規則DLL共享MFC DLL(或MFC擴展DLL),那么將總是默認使用EXE的資源
            3、因此應用程序需要通過資源模塊的切換來找到正確的資源。如果應用程序需要來自于DLL的資源,就應將資源模塊句柄指定為DLL的模塊句柄;如果需要EXE文件中包含的資源,就應將資源模塊句柄指定為EXE的模塊句柄。

            解決辦法:

            1、在DLL中改進:

            方法1。

            // in DLL
            void CDLL::ShowDlg(void)
            {
                   AFX_MANAGE_STATE(AfxGetStaticModuleState());
                   CDialog dlg(IDD_DLL_DIALOG); //打開ID為2000的對話框
                   dlg.DoModal();
            }

            注:AFX_MANAGE_STATE(AfxGetStaticModuleState());一定是作為接口函數的第一條語句。
                   其功能是在棧上(這意味著其作用域是局部的)創建一個AFX_MODULE_STATE類的實例,并將其指   針pModuleState返回。
                   AFX_MODULE_STATE類利用其構造函數和析構函數進行存儲模塊狀態現場及恢復現場的工作。
                   該宏用于將pModuleState設置為當前的有效模塊狀態。當離開該宏的作用域時(也就離開了pModuleState所指棧上對象的作用域),先前的模塊狀態將由類AFX_MODULE_STATE的析構函數恢復。(即自動恢復)

            方法2。

            // in DLL
            void CDLL::ShowDlg(void)
            {
                   HINSTANCE save_hInstance = AfxGetResourceHandle();
                   AfxSetResourceHandle(theApp.m_hInstance);
                   CDialog dlg(IDD_DLL_DIALOG); //打開ID為2000的對話框
                   dlg.DoModal();
                   AfxSetResourceHandle(save_hInstance);    
            }

            注:AfxGetResourceHandle:獲取當前資源模塊句柄;AfxSetResourceHandle:設置程序目前要使用的資源模塊句柄。
                   同方法1比較,方法2能夠靈活地設置程序的資源模塊句柄,而方法1則只能在DLL接口函數退出的時候才會恢復模塊句柄。

            2、在應用程序中改進:

            // in EXE
            void CEXE::OnButtonClick()
            {
                   HINSTANCE exe_hInstance = GetModuleHandle(NULL);
                  HINSTANCE dll_hInstance = GetModuleHandle("SharedDll.dll");
                  AfxSetResourceHandle(dll_hInstance); //切換狀態
                   ShowDlg();
                  AfxSetResourceHandle(exe_hInstance); //恢復狀態
            }

            注:使用狀態切換的情況:當DLL導出函數包含MFC資源、類或者需要創建窗口時。

            track: http://tech.163.com/06/0316/10/2CB1FGNG0009159F_4.html
            轉自

            http://blog.sina.com.cn/s/blog_62bb83b10100jbdj.html

            posted on 2013-03-22 14:58 zhangmo 閱讀(2719) 評論(0)  編輯 收藏 引用
             
            中文字幕乱码久久午夜| 久久中文字幕人妻丝袜| 男女久久久国产一区二区三区| 久久久久亚洲AV片无码下载蜜桃| 亚洲αv久久久噜噜噜噜噜| 国产成人精品久久一区二区三区 | 国产精品美女久久福利网站| 午夜精品久久久久| 1000部精品久久久久久久久| 国产精品久久久久久久午夜片| 亚洲国产精品久久久天堂| 国产精品美女久久久| 亚洲欧美国产精品专区久久| 丁香五月综合久久激情| 久久乐国产综合亚洲精品| 久久99中文字幕久久| 久久国产高潮流白浆免费观看| 久久精品成人欧美大片| 久久w5ww成w人免费| 一级a性色生活片久久无| 久久久久久免费一区二区三区| 久久99精品国产麻豆| 久久精品中文字幕无码绿巨人| 国产成人精品久久综合| 浪潮AV色综合久久天堂| 狠狠色噜噜色狠狠狠综合久久| 久久国产福利免费| 久久亚洲中文字幕精品一区四 | 久久精品国产2020| 香港aa三级久久三级老师2021国产三级精品三级在 | 亚洲&#228;v永久无码精品天堂久久 | 精品久久久久久亚洲精品| 日韩欧美亚洲综合久久| 久久久久亚洲精品中文字幕| 97久久综合精品久久久综合| 亚洲精品乱码久久久久久| 久久久久久免费视频| 国产成年无码久久久免费| 亚洲午夜久久久久久久久电影网| 久久热这里只有精品在线观看| 久久亚洲sm情趣捆绑调教|