• <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>
            隨筆-60  評(píng)論-262  文章-1  trackbacks-0

            http://kruglinski.blogchina.com/4609030.html

            以一個(gè)DropTarget為例,我們都知道在MFC里有COleDropTarget實(shí)現(xiàn)OLE拖放目標(biāo)端非常容易,
            缺點(diǎn) MFC 太臃腫近八年沒有更新過了, 而且功能類與窗體類分離, 代碼不夠緊湊, 那么在
            ATL/WTL 中要實(shí)現(xiàn) DropTarget 也是非常的容易的, 而且更碼更加緊湊完美, 完美的有點(diǎn)變
            態(tài),呵呵!

            我覺得ATL小組的人曾經(jīng)一定是些匯編語言狂熱者,因?yàn)锳TL的運(yùn)行效率真的是太高了,非常高,
            難以想象的高.(如果你偏不信可以用WTL向?qū)б粋€(gè)空工程,Release編

            譯后再用反匯編工具反一下, 看看生成的代碼質(zhì)量如何, 特別是使用 VC6 以后版本的 VC++
            編譯器編譯)

            首先向?qū)б粋€(gè)空的不帶工具欄和狀態(tài)欄的 WTL SDI 工程, Viwe type 選擇 Edit 這樣我們
            會(huì)得到一個(gè)類似 Notepad 界面的程序, 然后分成 6 步完成拖放支持

            1.在 stdafx.h 里加上以下語句:
             
            #include <atlcom.h>
             
            意義是使用 ATL 的 COM 支持類, 包括 CComObject, CComPtr 等
             
            2.在WinMain里:
            在 ::CoInitialize(NULL) 語句后加入以下語句
            ::OleInitialize(NULL);
            以及在
            ::CoUninitialize()語句前加入以下語句
            ::OleUninitialize();
             
            意義是在單線程套間中初使化 COM 庫, 初使化后便可使用以下功能
            a,剪貼板(Clipboard)
            b,拖與放(Drag & Drop)
            c,對(duì)象連接與嵌入(Object linking and embedding,OLE)
            d,就地激活(In-place activation)
             
            我原來一直以為使用 CoInitialize 就可以了, 可我調(diào)用 RegisterDragDrop 總是失敗, 并
            返回 E_OUTOFMEMORY, 直到我仔細(xì)看了函數(shù)說明看到下面這句話:
             
            Note 
            If you use CoInitialize or CoInitializeEx instead of OleInitialize to initialize
            COM, RegisterDragDrop will always return an E_OUTOFMEMORY error.
             
            很多時(shí)候 bug 都是因?yàn)椴蛔屑?xì)產(chǎn)生的, 呵呵!
             
            3. 將 CComObjectRoot 和 IDropTarget 加入 CMainFrame 的派生列表
             
            class CMainFrame : public CFrameWindowImpl<CMainFrame>,...,
             public CComObjectRootEx<CComSingleThreadModel>,public IDropTarget
             
            4.定義標(biāo)準(zhǔn)DropTarget方法
             
            在MainFrm.h的CMainFrm的類定義中定義標(biāo)準(zhǔn)的IDropTarget方法:
             
            STDMETHOD(DragEnter)(IDataObject * pDataObject,DWORD grfKeyState,POINTL pt,DWORD * pdwEffect);
            STDMETHOD(DragOver)(DWORD grfKeyState,POINTL pt,DWORD * pdwEffect);
            STDMETHOD(DragLeave)();
            STDMETHOD(Drop)(IDataObject * pDataObject,DWORD grfKeyState,POINTL pt,DWORD * pdwEffect);
             
            并在實(shí)現(xiàn)文件MainFrm.cpp實(shí)現(xiàn)它們.
             
            FORMATETC fe={0};
            STDMETHODIMP CMainFrame::DragEnter(IDataObject * pDataObject,DWORD grfKeyState,POINTL pt,DWORD * pdwEffect)
            {
             CComPtr<IEnumFORMATETC> pEnum;
             pDataObject->EnumFormatEtc(DATADIR_GET,&pEnum);
             while(pEnum->Next(1,&fe,NULL)==NO_ERROR)
             {
              if(fe.cfFormat==CF_TEXT)
              {
               *pdwEffect=DROPEFFECT_COPY;
               break;
              }
             }
             return S_OK;
            }
            STDMETHODIMP CMainFrame::DragOver(DWORD grfKeyState,POINTL pt,DWORD * pdwEffect)
            {
             *pdwEffect=DROPEFFECT_COPY;
             return S_OK;
            }
             
            STDMETHODIMP CMainFrame::DragLeave()
            {
             return S_OK;
            }
             
            STDMETHODIMP CMainFrame::Drop(IDataObject * pDataObject,DWORD grfKeyState,POINTL pt,DWORD * pdwEffect)
            {
             STGMEDIUM stg={0};
             pDataObject->GetData(&fe,&stg);
             LPCTSTR lpData=(LPCTSTR)GlobalLock(stg.hGlobal);
             
             m_view.SetWindowText(lpData);
             
             GlobalUnlock(stg.hGlobal);
             ReleaseStgMedium(&stg);
             
             *pdwEffect=DROPEFFECT_COPY;
             return S_OK;
            }
             
             
            5.定義 COM 映射表
             
            在MainFrm.h的CMainFrm的類定義中加入下面幾句:
            BEGIN_COM_MAP(CMainFrame)
                COM_INTERFACE_ENTRY(IDropTarget)
            END_COM_MAP()
             
            6.注冊(cè)和注銷
            在WM_CREATE消息的Handler OnCreate中注冊(cè)
            RegisterDragDrop(m_hWnd,this);
            在WM_CLOSE消息的Handler OnClose中注銷
            RevokeDragDrop(m_hWnd);
             
            現(xiàn)在差不多已經(jīng)完成了, 在這里不要問怎么沒有見到你寫 AddRef, Release, QueryInterface,
            ATL 為我們提供了非常高效且多線程安全的實(shí)現(xiàn), 我們要做的只是實(shí)現(xiàn)不同的接口, COM 對(duì)
            象的生存周期管理是基于 "在堆中生成對(duì)象" 的假設(shè) (否則還 AddRef, Release干什么).
            這里我們的主窗體對(duì)象也將在堆中生成.
             
            修改一下 Run 函數(shù), 像下面這樣來生成主窗體
             
            int Run(LPTSTR /*lpstrCmdLine*/ = NULL, int nCmdShow = SW_SHOWDEFAULT)
            {
             CMessageLoop theLoop;
             _Module.AddMessageLoop(&theLoop);
             
             //一個(gè)COM對(duì)象指針
             CComObject<CMainFrame> *pMainFrm=NULL;
             //在堆上分配對(duì)象
             CComObject<CMainFrame>::CreateInstance(&pMainFrm);
             
             if(pMainFrm->CreateEx() == NULL)
             {
              ATLTRACE(_T("Main window creation failed!\n"));
              return 0;
             }
             
             //增加引用計(jì)數(shù),COM的老規(guī)矩,不用我多說
             pMainFrm->AddRef();
             pMainFrm->ShowWindow(nCmdShow);
             
             //開始消息循環(huán),Release編譯后都是展開的
             //直接調(diào)用GetMessage,TranslateMessage.....
             int nRet = theLoop.Run();
             
             //減少引用計(jì)數(shù),COM的老規(guī)矩,這里的計(jì)數(shù)為1,
             //調(diào)用Release后對(duì)象自動(dòng)析構(gòu)
             pMainFrm->Release();
             
             _Module.RemoveMessageLoop();
             return nRet;
            }
             
            運(yùn)行一下, 可以從 WinWord 里拖放文本到這個(gè)小程序里.
             
            從注冊(cè)時(shí)的 RegisterDragDrop(m_hWnd, this) 調(diào)用可以看到窗口對(duì)象和 COM 對(duì)象完美的
            融合到了一起, 在 COM 方法中可以很方便的與其它組成部分交互, 比如 Drop 方法中的
            m_view.SetWindowText(lpData)調(diào)用. 為什么 this 可以自動(dòng)轉(zhuǎn)成 IDropTarget 指針呢,
            因?yàn)槲覀兊?CMainFrame 繼承了 IDropTarget抽像類, 所以按照面向?qū)ο蟮挠^念來看
            CMainFrame 類 "是一個(gè)" IDropTarget 類.
             
            其實(shí)在 CMainFrame 中只有少數(shù)接口繼承來的函數(shù)會(huì)生成函數(shù)體, 其它的如消息處理過程
            OnCreate, OnClose 等如果代碼不是很多,最后都會(huì)內(nèi)聯(lián)到一個(gè)叫做 ProcessWindowMessage
            的函數(shù)中, 由一個(gè)大大的 switch 來處理.
             
            再優(yōu)化一下, 我們?nèi)绻涌诶^承的層次太多(其實(shí)這個(gè)例子里不多), 便會(huì)生成龐大虛函數(shù)表,
            從而影響性能, 這是一些偏執(zhí)狂對(duì) C++ 一直都指責(zé)的地方, 給 CMainFrame 加上
            ATL_NO_VTABLE(__declspec(novtable)) 定義, 這樣如果是從 CMainFrame 繼承, 也只到最
            后的實(shí)現(xiàn)類才生成虛函數(shù)表.
             
            使用默認(rèn)的 Release 編譯選項(xiàng) (最大化速度優(yōu)先) 編譯生成的 exe 只有 36K, 而一個(gè)
            Win32 Application 向?qū)傻膸Т翱诳展こ叹幾g后都有 40K 了,知道為什么嗎? 呵呵, 因
            為 ATL 在 Release 編譯時(shí)用更小更快的啟動(dòng)代碼而不是 _WinMainCRTStartup 之類的東東,
            并且不使用 CRT, C++ RTTI, C++ 異常處理,以及自定義了一套 malloc, new, free, delete
            之類的函數(shù)和運(yùn)算符 (這不就是使用 C++ 語法在寫 C 程序嘛! 或者說比 C 更高效, 連 CRT
            都不用), 如果再加上
             
            /opt:nowin98, /align 之類的指示便會(huì)更小.
             
            總之, 在我的腦海里: ATL+WTL=快速小巧且界面漂亮的程序,:-)
             
            其實(shí)這些優(yōu)化在機(jī)器性能猛增的今天已經(jīng)變的微不足道, 畢竟開發(fā)工作不全都是在處理海量
            的多媒體數(shù)據(jù). 現(xiàn)在用 MFC 的人都少了, 更不用說 ATL 這樣出力不討好的東東, 大家都在
            搞 .NET 用 C# 之類的東東, 我一直很不喜歡解釋執(zhí)行的東東, 可往后每臺(tái)電腦都有了
            .NET Framework, 就不存在托管代碼和本地代碼之分了, 反正發(fā)布的軟件在哪臺(tái)電腦里都能
            運(yùn)行, 就像你用 Windows API 來寫程序, 系統(tǒng)中已經(jīng)帶好了相關(guān)的 DLL, 以后用 .NET 框
            架來寫程序, 系統(tǒng)中也有 .NET 程序的運(yùn)行時(shí)環(huán)境,用戶不會(huì)管你是真編譯的還是解釋執(zhí)行的
            或是你少用了幾個(gè)時(shí)鐘周期, 發(fā)現(xiàn)自己需要轉(zhuǎn)變一下那種偏執(zhí)狂的想法. 也許什么時(shí)候我應(yīng)
            該接受 C# 或是 Java.....

            posted on 2008-03-08 14:41 free2000fly 閱讀(2022) 評(píng)論(3)  編輯 收藏 引用

            評(píng)論:
            # re: 用 ATL/WTL 中實(shí)現(xiàn) DropTarget(轉(zhuǎn)載) 2008-08-29 17:25 | xyj
            朋友,有做過這方面例子嗎?能不能請(qǐng)教下啊  回復(fù)  更多評(píng)論
              
            # re: 用 ATL/WTL 中實(shí)現(xiàn) DropTarget(轉(zhuǎn)載) 2008-08-30 19:20 | free2000fly
            @xyj
            你夠懶, 照葫蘆畫瓢一路下來就能得到一個(gè)例子代碼. 自求多福吧.  回復(fù)  更多評(píng)論
              
            # re: 用 ATL/WTL 中實(shí)現(xiàn) DropTarget(轉(zhuǎn)載) 2009-07-17 16:57 | flyingbugs
            兄弟,寫的不錯(cuò)。加個(gè)msn吧:
            flyingbugs@live.cn  回復(fù)  更多評(píng)論
              

            只有注冊(cè)用戶登錄后才能發(fā)表評(píng)論。
            網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


            久久er国产精品免费观看2| 亚洲性久久久影院| 久久久免费精品re6| 精品国产99久久久久久麻豆| 久久SE精品一区二区| 久久99热这里只有精品国产| 国产精品国色综合久久| 国产高潮国产高潮久久久91 | 2020久久精品亚洲热综合一本| 国产精品女同一区二区久久| 国产精品一区二区久久精品涩爱| 久久99精品久久久久久hb无码| 精品久久久久久久久久中文字幕 | 91精品国产91久久| 亚洲精品国精品久久99热| 国产麻豆精品久久一二三| 国产成人99久久亚洲综合精品| 久久久无码精品亚洲日韩蜜臀浪潮| 欧美熟妇另类久久久久久不卡| 精品久久久久一区二区三区| 午夜精品久久久久久99热| 亚洲国产精久久久久久久| 无码人妻少妇久久中文字幕蜜桃| 99久久精品免费观看国产| 久久亚洲私人国产精品| 国产69精品久久久久APP下载| 国产精品VIDEOSSEX久久发布 | 久久精品无码专区免费| 国产精品久久久久久久久免费| 一本久久综合亚洲鲁鲁五月天亚洲欧美一区二区| 日本人妻丰满熟妇久久久久久| 国产毛片久久久久久国产毛片| 久久国产欧美日韩精品| 亚洲精品无码久久一线| 久久天天婷婷五月俺也去| 欧美一级久久久久久久大片| 久久精品成人欧美大片| 精品久久久久久无码人妻热| 久久天天躁狠狠躁夜夜2020老熟妇| 久久精品国产亚洲网站| 久久精品成人国产午夜|