• <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>
            隨筆-341  評(píng)論-2670  文章-0  trackbacks-0
                GacUI為了實(shí)現(xiàn)把界面序列化和反序列化到XML,必然要有類似反射一樣的功能。但是C++卻沒有反射,現(xiàn)在想到的方法就是,把編譯后的pdb文件拿出來(lái)。因?yàn)榭丶皇悄0孱悾詳?shù)據(jù)都可以直接獲取。pdb文件包含了所有函數(shù)的信息,還有被實(shí)例化后的模板類和模板函數(shù)的信息。因此只需要使用IDiaDataSource(Visual Studio提供的COM組件)讀取pdb的類聲明之后,把信息整理并輸出到一個(gè)xml里面,然后就可以用C#編寫linq to xml的程序去分析并生成支持C++反射的一系列周邊代碼了。這樣就自動(dòng)讓C++其中一部分必要的類獲得反射的功能,代價(jià)就是每一次修改完代碼之后,要記得非人肉地更新自動(dòng)生成的代碼。

                不過(guò)為了更加形象的展示pdb的內(nèi)容,我使用GacUI的帶Virtual Mode的TreeView打開pdb填充。這里面有兩個(gè)view,第一個(gè)是pdb,第二個(gè)是整理后的class view。顯示pdb的GuiTreeView控件展示了如何通過(guò)提供一個(gè)數(shù)據(jù)源,從而實(shí)現(xiàn)“展開的時(shí)候再?gòu)膒db文件里面讀取信息”的技術(shù)。而class view則是通過(guò)提供一個(gè)數(shù)據(jù)源來(lái)將一個(gè)文件中的xml讀取到內(nèi)存并顯示出來(lái),但是避免new那些暫時(shí)還不需要顯示出來(lái)的TreeViewNode對(duì)象。代碼放在Vczh Library++ 3.0(Candidate\GUI\GUIDemo\GUIDemo.sln)。現(xiàn)在先上圖:





                解析PDB的關(guān)鍵代碼在DumpPDB.cpp文件中,大家只需要下載代碼并閱讀即可。所有的內(nèi)容都可以從MSDN搜索IDiaDataSource獲得,但是運(yùn)行的話則需要有這個(gè)COM組件,一般要求安裝Visual Studio。下面解釋一下一段C++代碼。這是上面那個(gè)按鈕的回調(diào)函數(shù)。這個(gè)回調(diào)函數(shù)做了下面幾件事情
                1、將Button和TagPage都Disable
                2、利用線程池異步將PDB的內(nèi)容保存到XML文件中(一秒鐘)
                3、第2步完成之后,發(fā)一個(gè)消息回到GUI線程,自動(dòng)顯示第二個(gè)TagPage
                4、異步將XML讀取到內(nèi)存。在這里我沒有使用延遲讀取技術(shù),所以我直接創(chuàng)建了大約幾百萬(wàn)個(gè)字符串,需要五秒鐘
                5、第4步完成之后,發(fā)一個(gè)消息回到GUI線程嗎,將創(chuàng)建好的內(nèi)存中的XML格式顯示在TreeView里

                這些異步操作來(lái)往十分復(fù)雜,但是借助C++0x就可以描述得十分清晰。GacUI的實(shí)現(xiàn)并沒有使用C++0x,但是仍然可以為使用C++0x的那部分用戶提供一些更加優(yōu)化的接口。因此這些復(fù)雜的步驟最后就寫成了:

            buttonDump->Clicked.AttachLambda([=](GuiGraphicsComposition* sender, GuiEventArgs& arguments)
            {
                INativeController* controller=GetCurrentController();
                tabControl->GetPages()[0]->GetContainer()->SetEnabled(false);
                buttonDump->SetEnabled(false);
                buttonDump->SetText(L"Dumping...");
                buttonDump->GetRelatedControlHost()->GetBoundsComposition()->SetAssociatedCursor(controller->GetSystemCursor(INativeCursor::LargeWaiting));

                ThreadPoolLite::QueueLambda([=]()
                {
                    dumppdb::DumpPdbToXml(diaSymbol, L"..\\Debug\\GuiDemo.xml");
                    GetApplication()->InvokeLambdaInMainThread([=]()
                    {
                        tabControl->GetPages()[0]->GetContainer()->SetEnabled(true);
                        tabControl->SetSelectedPage(tabControl->GetPages()[1]);
                        buttonDump->SetText(L"Loading GuiDemo.xml in the class view...");

                        ThreadPoolLite::QueueLambda([=]()
                        {
                            FileStream fileStream(L"..\\Debug\\GuiDemo.xml", FileStream::ReadOnly);
                            CacheStream cacheStream(fileStream, 1048576);
                            BomDecoder decoder;
                            DecoderStream decoderStream(cacheStream, decoder);
                            StreamReader reader(decoderStream);
                            Ptr<TreeElement> xml=LoadXmlRawDocument(reader).Cast<TreeElement>();

                            GetApplication()->InvokeLambdaInMainThreadAndWait([=]()
                            {
                                buttonDump->SetText(L"GuiDemo.xml dumpped.");
                                buttonDump->GetRelatedControlHost()->GetBoundsComposition()->SetAssociatedCursor(controller->GetDefaultSystemCursor());

                                GuiTreeView* treeControl=new GuiTreeView(new win7::Win7TreeViewProvider, CreateProviderFromXml(xml));
                                treeControl->GetBoundsComposition()->SetAlignmentToParent(Margin(0, 0, 0, 0));
                                treeControl->SetVerticalAlwaysVisible(false);
                                treeControl->SetHorizontalAlwaysVisible(false);
                                tabControl->GetPages()[1]->GetContainer()->GetContainerComposition()->AddChild(treeControl->GetBoundsComposition());
                            });
                        });
                    });
                });
            });


                buttonDump->Clicked.AttachLambda的意思是將一個(gè)滿足C++0x標(biāo)準(zhǔn)的lambda表達(dá)式當(dāng)成一個(gè)事件處理程序綁定到一個(gè)時(shí)間上。ThreadPoolLite::QueueLambda則是將一個(gè)lambda表達(dá)式放進(jìn)Windows內(nèi)核實(shí)現(xiàn)的內(nèi)存池進(jìn)行異步調(diào)用。GetApplication()->InvokeLambdaInMainThread(AndWait)則是在別的線程里將一個(gè)lambda表達(dá)式放到GUI線程(一般是主線程)中運(yùn)行。如果調(diào)用了Wait的版本,則這個(gè)函數(shù)會(huì)一直等到該lambda表達(dá)式在主線程執(zhí)行完了才會(huì)返回。如果大家關(guān)心實(shí)現(xiàn)的話,可以去Candidate\GUI\GUI\NativeWindow\Windows\WinNativeWindow.cpp文件里查看。

                大家可以想象,在古老的不支持lambda表達(dá)式的C++版本里面,要實(shí)現(xiàn)這個(gè)過(guò)程,這個(gè)函數(shù)將被拆散成多少函數(shù)。為了傳遞很多復(fù)雜的對(duì)象,要寫多少個(gè)臨時(shí)的struct,new多少內(nèi)存碎片才能將異步回調(diào)函數(shù)的參數(shù)做成Windows所希望的DWORD(__stdcall*)(void*)格式。為了把一部分事情放回到GUI線程做(我們都知道GUI庫(kù)不值得為了線程安全而做很多浪費(fèi)性能的事情),得實(shí)現(xiàn)多少私有的Win32消息,subclass多少東西才能最終做到。這一切在GacUI中都簡(jiǎn)化了。

                接下來(lái)將會(huì)研究如何利用pdb里面的信息讓跟GacUI有關(guān)的對(duì)象支持反射的具體細(xì)節(jié)。元旦就先休息了,啊哈哈哈。
            posted on 2011-12-30 04:12 陳梓瀚(vczh) 閱讀(7625) 評(píng)論(21)  編輯 收藏 引用 所屬分類: GacUI

            評(píng)論:
            # re: GacUI Demo:PDB Viewer(分析pdb文件并獲取C++類聲明的詳細(xì)內(nèi)容) 2011-12-30 07:32 | 空明流轉(zhuǎn)
            # re: GacUI Demo:PDB Viewer(分析pdb文件并獲取C++類聲明的詳細(xì)內(nèi)容)[未登錄] 2011-12-30 08:13 | 春秋十二月
            準(zhǔn)備有空研究下lambda  回復(fù)  更多評(píng)論
              
            # re: GacUI Demo:PDB Viewer(分析pdb文件并獲取C++類聲明的詳細(xì)內(nèi)容) 2011-12-30 08:59 | foxriver
            前幾天被老版本pdb格式折磨的死去活來(lái),由于源文件多,大于64M報(bào)錯(cuò),把pdb嵌入dll里也報(bào)錯(cuò)。無(wú)奈只能用line number only來(lái)調(diào)試,郁悶的不行。  回復(fù)  更多評(píng)論
              
            # re: GacUI Demo:PDB Viewer(分析pdb文件并獲取C++類聲明的詳細(xì)內(nèi)容) 2011-12-30 09:01 | 陳梓瀚(vczh)
            @foxriver
            重新編譯一次吧……  回復(fù)  更多評(píng)論
              
            # re: GacUI Demo:PDB Viewer(分析pdb文件并獲取C++類聲明的詳細(xì)內(nèi)容) 2011-12-30 17:38 | 戰(zhàn)魂小筑
            難怪MS要推F#,現(xiàn)在函數(shù)編程那么熱, 內(nèi)核線程池啊..

            終于有人分析pdb了,但是看到com和必裝visualstudio, 看來(lái)沒戲了,還不如自己寫個(gè)分析器做反射來(lái)的快  回復(fù)  更多評(píng)論
              
            # re: GacUI Demo:PDB Viewer(分析pdb文件并獲取C++類聲明的詳細(xì)內(nèi)容) 2011-12-30 18:26 | 陳梓瀚(vczh)
            @戰(zhàn)魂小筑
            pdb的二進(jìn)制復(fù)雜到死,對(duì)此深表同情。話說(shuō)一個(gè)只給windows程序員(因?yàn)槭莗db)使用的工具,你完全可以認(rèn)為對(duì)方已經(jīng)安裝了visual studio。這樣你就可以借助msdn在半天內(nèi)搞定這個(gè)事情了。  回復(fù)  更多評(píng)論
              
            # re: GacUI Demo:PDB Viewer(分析pdb文件并獲取C++類聲明的詳細(xì)內(nèi)容) 2011-12-30 20:24 | ooseven
            兩個(gè)問(wèn)題:
            1、pdb文件不同的vs版本有沒有兼容性問(wèn)題?
            2、可不可以有選擇只針對(duì)有需要序列化的UI類才導(dǎo)出到xml?

            這個(gè)方案很有趣,但是,要人工小心的維護(hù),意味著只要UI類里增加了一個(gè)變量,都需要重新導(dǎo)出!  回復(fù)  更多評(píng)論
              
            # re: GacUI Demo:PDB Viewer(分析pdb文件并獲取C++類聲明的詳細(xì)內(nèi)容) 2011-12-30 20:49 | ooseven
            我覺得這個(gè)方法很復(fù)雜,而且會(huì)增加不穩(wěn)定性與人工維護(hù)出錯(cuò)幾率,最好的方案是增加一個(gè)CUI_XMLSerializationBase類,在里面實(shí)現(xiàn)類的反射機(jī)制,然后所有需要序列化的UI類都需要從這個(gè)類繼承,這樣既簡(jiǎn)單,又避免手工維護(hù),又有良好的兼容性。  回復(fù)  更多評(píng)論
              
            # re: GacUI Demo:PDB Viewer(分析pdb文件并獲取C++類聲明的詳細(xì)內(nèi)容) 2011-12-30 23:52 | 陳梓瀚(vczh)
            @ooseven
            這,編譯后運(yùn)行一個(gè)bat,是肯定不會(huì)忘記的。如果還忘記,那就把這個(gè)bat加入到“編譯后運(yùn)行”里面,每次都自動(dòng)運(yùn)行,肯定忘不了。不知道你是怎么覺得復(fù)雜的。

            而且我那個(gè)xml是不會(huì)放到程序里面的。流程是這樣的
            1:編譯一次獲得pdb
            2:生成xml
            3:從xml生成代碼
            4:代碼合并進(jìn)去再編譯一次

            自始自終pdb是新的,xml是新的,代碼是新的,而且編譯出來(lái)的東西不需要帶著xml文件也可以運(yùn)行。所以你的問(wèn)題的答案就是

            1:GacUI的代碼全都有,所以你是不可能需要用不同的vs來(lái)產(chǎn)生同一個(gè)exe的兩個(gè)部分的……
            2:導(dǎo)出xml不管,應(yīng)該在生成代碼的時(shí)候做過(guò)濾。  回復(fù)  更多評(píng)論
              
            # re: GacUI Demo:PDB Viewer(分析pdb文件并獲取C++類聲明的詳細(xì)內(nèi)容) 2011-12-31 00:48 | 春秋十一月
            這流程沒必要那么復(fù)雜,完全可以通過(guò)hookapi的方式,修改cl.exe讀入的cpp文件,生成一個(gè)臨時(shí)內(nèi)存文件提供給編譯器。只要源代碼行數(shù)不變,完全沒有影響的。就類似qt moc工具。

            pdb解析早先有國(guó)外牛人寫過(guò),用于scenedemo. 后來(lái)微軟放出了vc2010 pdb com api, codeproject就有人做了一個(gè),現(xiàn)在也漸漸流行開來(lái)。只是個(gè)人感覺,還是讀取文本類型的.map要簡(jiǎn)單多了。
              回復(fù)  更多評(píng)論
              
            # re: GacUI Demo:PDB Viewer(分析pdb文件并獲取C++類聲明的詳細(xì)內(nèi)容) 2011-12-31 01:04 | 陳梓瀚(vczh)
            @春秋十一月
            hack的事情,一般不要做。免得依賴于人家的bug。而且這不僅僅是一個(gè)運(yùn)行時(shí)的問(wèn)題。沒有編譯期反射,你怎么寫程序自動(dòng)替你寫dll外殼,怎么讓反射跟dll的接口長(zhǎng)一個(gè)樣子,怎么做腳本引擎的插件?  回復(fù)  更多評(píng)論
              
            # re: GacUI Demo:PDB Viewer(分析pdb文件并獲取C++類聲明的詳細(xì)內(nèi)容) 2011-12-31 08:15 | 春秋十一月
            hack是人類進(jìn)步之階梯。編程要方便,首先要改善編譯器。比如apple想加一個(gè)thread block語(yǔ)法,c++不支持怎么辦,就改gcc, gcc不滿足要求,就用clang來(lái)替換,要不知足,這才是thinking in apple way.  回復(fù)  更多評(píng)論
              
            # re: GacUI Demo:PDB Viewer(分析pdb文件并獲取C++類聲明的詳細(xì)內(nèi)容) 2011-12-31 09:20 | 陳梓瀚(vczh)
            @春秋十一月
            所以apple才人頭數(shù)那么少。做工具是不能這么干的,我又不是在賣那些虛無(wú)飄渺的理念。  回復(fù)  更多評(píng)論
              
            # re: GacUI Demo:PDB Viewer(分析pdb文件并獲取C++類聲明的詳細(xì)內(nèi)容) 2012-01-01 07:23 | some
            等于說(shuō)你重新造了一個(gè)界面庫(kù)的"輪子",更積極的意義是什么呢?  回復(fù)  更多評(píng)論
              
            # re: GacUI Demo:PDB Viewer(分析pdb文件并獲取C++類聲明的詳細(xì)內(nèi)容) 2012-01-01 07:56 | 陳梓瀚(vczh)
            @some
            不僅Windows下面C++從此有了GPU加速的GUI,而且造完輪子后我變得更厲害了。難道這個(gè)意義不是相當(dāng)?shù)姆e極嗎?  回復(fù)  更多評(píng)論
              
            # re: GacUI Demo:PDB Viewer(分析pdb文件并獲取C++類聲明的詳細(xì)內(nèi)容) 2012-01-03 17:01 | netivs
            你文章里面說(shuō)“pdb文件包含了所有函數(shù)的信息,還有被實(shí)例化后的模板類和模板函數(shù)的信息。”,請(qǐng)問(wèn)有沒有工具可以從pdb文件里面獲取到所有的函數(shù)聲明(包括函數(shù)名稱、返回值、參數(shù)類型等)?有的話麻煩給個(gè)鏈接。謝謝!  回復(fù)  更多評(píng)論
              
            # re: GacUI Demo:PDB Viewer(分析pdb文件并獲取C++類聲明的詳細(xì)內(nèi)容) 2012-01-03 22:12 | 陳梓瀚(vczh)
            @netivs
            我這不就寫了一個(gè)嗎?快下載。  回復(fù)  更多評(píng)論
              
            # re: GacUI Demo:PDB Viewer(分析pdb文件并獲取C++類聲明的詳細(xì)內(nèi)容) 2012-01-11 03:32 | ArthasLee
            @陳梓瀚(vczh)
            支持造輪子讓自己變得更厲害黨;
              回復(fù)  更多評(píng)論
              
            # re: GacUI Demo:PDB Viewer(分析pdb文件并獲取C++類聲明的詳細(xì)內(nèi)容) 2012-01-12 08:09 | Scan
            在云風(fēng)那邊也是一天到晚看到輪子黨,張口必然輪子,真受不了!
            輪子造得越多越牛逼!
            現(xiàn)在的新人程序,本來(lái)編碼經(jīng)驗(yàn)就少,如果不多造輪子,必然迅速得淪為控件黨!  回復(fù)  更多評(píng)論
              
            # re: GacUI Demo:PDB Viewer(分析pdb文件并獲取C++類聲明的詳細(xì)內(nèi)容) 2012-01-12 08:27 | 陳梓瀚(vczh)
            @Scan
            我一直都認(rèn)為,用業(yè)余時(shí)間造輪子,是很值得的。  回復(fù)  更多評(píng)論
              
            # re: GacUI Demo:PDB Viewer(分析pdb文件并獲取C++類聲明的詳細(xì)內(nèi)容) 2015-06-05 05:59 | dzw
            編譯不過(guò)  回復(fù)  更多評(píng)論
              
            久久综合给久久狠狠97色| 日本福利片国产午夜久久| 久久久久国产精品三级网| 久久人人爽人人爽人人片AV麻豆 | 久久精品无码一区二区无码| 久久人人爽人人爽人人片av高请 | 久久久久国产一级毛片高清板| 亚洲va久久久久| 久久久国产精华液| 久久香综合精品久久伊人| 欧美熟妇另类久久久久久不卡| 久久天天躁狠狠躁夜夜躁2O2O| 国产91色综合久久免费| 亚洲国产成人久久一区WWW| 久久久女人与动物群交毛片 | 欧美国产成人久久精品| 99久久婷婷国产一区二区| 中文字幕精品无码久久久久久3D日动漫| 777午夜精品久久av蜜臀| 精品熟女少妇aⅴ免费久久| 久久久久亚洲AV成人片| 亚洲欧美日韩精品久久亚洲区 | 久久精品一区二区三区AV| 欧美伊香蕉久久综合类网站| 久久夜色精品国产噜噜噜亚洲AV| 欧洲性大片xxxxx久久久| 久久66热人妻偷产精品9| 伊人久久亚洲综合影院| 久久久久18| 99久久99久久精品国产片| 国产亚洲色婷婷久久99精品| 久久妇女高潮几次MBA| 中文字幕无码久久人妻| 日韩十八禁一区二区久久| 久久久久国产日韩精品网站| 精品无码久久久久久久动漫| 国产AⅤ精品一区二区三区久久| 亚洲国产二区三区久久| 久久99国产精品一区二区| 99精品伊人久久久大香线蕉| 国产午夜电影久久|