• <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>

            C++ Programmer's Cookbook

            {C++ 基礎(chǔ)} {C++ 高級(jí)} {C#界面,C++核心算法} {設(shè)計(jì)模式} {C#基礎(chǔ)}

            C#中重用c/c++舊模塊

             一、發(fā)生的背景
              在開發(fā)新項(xiàng)目中使用了新的語言開發(fā) C# 和新的技術(shù)方案 WEB Service,但是在新項(xiàng)目中,一些舊的模塊需要繼續(xù)使用,一般是采用 C 或 C++ 或 Delphi 編寫的,如何利用舊模塊對(duì)于開發(fā)人員來說,有三種可用方法供選擇:

              第一、將 C 或 C++ 函數(shù)用 C# 徹底改寫一遍,這樣整個(gè)項(xiàng)目代碼比較統(tǒng)一,維護(hù)也方便一些。但是盡管微軟以及某些書籍說,C# 和 C++ 如何接近,但是改寫起來還是很痛苦的事情,特別是 C++ 里的指針和內(nèi)存操作;

              第二、將 C 或 C++ 函數(shù)封裝成 COM,在 C# 中調(diào)用COM 比較方便,只是在封裝時(shí)需要處理 C 或 C++ 類型和 COM 類型之間的轉(zhuǎn)換,也有一些麻煩,另外COM 還需要注冊(cè),注冊(cè)次數(shù)多了又可能導(dǎo)致混亂;

              第三、將 C 或 C++ 函數(shù)封裝成動(dòng)態(tài)鏈接庫,封裝的過程簡單,工作量不大。因此我決定采用加載動(dòng)態(tài)鏈接庫的方法實(shí)現(xiàn),于是產(chǎn)生了在 C# 中如何調(diào)用自定義的動(dòng)態(tài)鏈接庫問題,我在網(wǎng)上搜索相關(guān)主題,發(fā)現(xiàn)一篇調(diào)用系統(tǒng) API 的文章,但是沒有說明如何解決此問題,在 MSDN 上也沒有相關(guān)詳細(xì)說明。基于此,我決定自己從簡單出發(fā),逐步試驗(yàn),看看能否達(dá)到自己的目標(biāo)。

              (說明一點(diǎn):我這里改寫為什么很怕麻煩,我改寫的代碼是變長加密算法函數(shù),代碼有600多行,對(duì)算法本身不熟悉,算法中指針和內(nèi)存操作太多,要想保證算法正確,最可行的方法就是少動(dòng)代碼,否則只要有一點(diǎn)點(diǎn)差錯(cuò),就不能肯定算法與以前兼容)

            二、技術(shù)實(shí)現(xiàn)


              下面看看如何逐步實(shí)現(xiàn)動(dòng)態(tài)庫的加載,類型的匹配,動(dòng)態(tài)鏈接庫函數(shù)導(dǎo)出的定義,這個(gè)不需要多說,大家參考下面宏定義即可:

              #define LIBEXPORT_API extern "C" __declspec(dllexport)
              第一步,我先從簡單的調(diào)用出發(fā),定義了一個(gè)簡單的函數(shù),該函數(shù)僅僅實(shí)現(xiàn)一個(gè)整數(shù)加法求和:

              LIBEXPORT_API int mySum(int a,int b){ return a+b;}
              C# 導(dǎo)入定義:


              public class RefComm
              {
              [DllImport("LibEncrypt.dll",
               EntryPoint=" mySum ",
               CharSet=CharSet.Auto,CallingConvention=CallingConvention.StdCall)]
               public static extern int mySum (int a,int b);
              }



              在C#中調(diào)用測試:

              int iSum = RefComm.mySum(2,3);
              運(yùn)行查看結(jié)果iSum為5,調(diào)用正確。第一步試驗(yàn)完成,說明在C#中能夠調(diào)用自定義的動(dòng)態(tài)鏈接庫函數(shù)。
            第二步,我定義了字符串操作的函數(shù)(簡單起見,還是采用前面的函數(shù)名),返回結(jié)果為字符串:

              LIBEXPORT_API char *mySum(char *a,char *b){sprintf(b,"%s",a); return a;}
              C# 導(dǎo)入定義:


              public class RefComm
              {
              [DllImport("LibEncrypt.dll",
               EntryPoint=" mySum ",
               CharSet=CharSet.Auto,
               CallingConvention=CallingConvention.StdCall)]
               public static extern string mySum (string a, string b);
              }


              在C#中調(diào)用測試:

              string strDest="";
              string strTmp= RefComm.mySum("12345", strDest);

              運(yùn)行查看結(jié)果 strTmp 為"12345",但是strDest為空。我修改動(dòng)態(tài)鏈接庫實(shí)現(xiàn),返回結(jié)果為串b:

              LIBEXPORT_API char *mySum(char *a,char *b){sprintf(b,"%s",a) return b;}

              修改 C# 導(dǎo)入定義,將串b修改為ref方式:


              public class RefComm
              {
              [DllImport("LibEncrypt.dll",
              EntryPoint=" mySum ",
              CharSet=CharSet.Auto,CallingConvention=CallingConvention.StdCall)]
              public static extern string mySum (string a, ref string b);
              }


              在C#中再調(diào)用測試:

              string strDest="";
              string strTmp= RefComm.mySum("12345", ref strDest);
              運(yùn)行查看結(jié)果 strTmp 和 strDest 均不對(duì),含不可見字符。再修改 C# 導(dǎo)入定義,將CharSet從Auto修改為Ansi:


              public class RefComm
              {
              [DllImport("LibEncrypt.dll",
               EntryPoint=" mySum ",
               CharSet=CharSet.Ansi,CallingConvention=CallingConvention.StdCall)]
               public static extern string mySum (string a, string b);
              }

            在C#中再調(diào)用測試:

              string strDest="";
              string strTmp= RefComm. mySum("12345", ref strDest);

              運(yùn)行查看結(jié)果 strTmp 為"12345",但是串 strDest 沒有賦值。第二步實(shí)現(xiàn)函數(shù)返回串,但是在函數(shù)出口參數(shù)中沒能進(jìn)行輸出。再次修改 C# 導(dǎo)入定義,將串b修改為引用(ref):


              public class RefComm
              {
              [DllImport("LibEncrypt.dll",
               EntryPoint=" mySum ",
               CharSet=CharSet.Ansi,CallingConvention=CallingConvention.StdCall)]
               public static extern string mySum (string a, ref string b);
              }


              運(yùn)行時(shí)調(diào)用失敗,不能繼續(xù)執(zhí)行。

              第三步,修改動(dòng)態(tài)鏈接庫實(shí)現(xiàn),將b修改為雙重指針:

              LIBEXPORT_API char *mySum(char *a,char **b){sprintf((*b),"%s",a); return *b;}

              C#導(dǎo)入定義:


              public class RefComm
              {
              [DllImport("LibEncrypt.dll",
               EntryPoint=" mySum ",
               CharSet=CharSet.Ansi,CallingConvention=CallingConvention.StdCall)]
               public static extern string mySum (string a, ref string b);
              }


              在C#中調(diào)用測試:

              string strDest="";
              string strTmp= RefComm. mySum("12345", ref strDest);

              運(yùn)行查看結(jié)果 strTmp 和 strDest 均為"12345",調(diào)用正確。第三步實(shí)現(xiàn)了函數(shù)出口參數(shù)正確輸出結(jié)果。
            第四步,修改動(dòng)態(tài)鏈接庫實(shí)現(xiàn),實(shí)現(xiàn)整數(shù)參數(shù)的輸出:

              LIBEXPORT_API int mySum(int a,int b,int *c){ *c=a+b; return *c;}
              
              C#導(dǎo)入的定義:


              public class RefComm
              {
              [DllImport("LibEncrypt.dll",
               EntryPoint=" mySum ",
               CharSet=CharSet.Ansi,CallingConvention=CallingConvention.StdCall)]
               public static extern int mySum (int a, int b,ref int c);
              }


              在C#中調(diào)用測試:

              int c=0;
              int iSum= RefComm. mySum(2,3, ref c);

              運(yùn)行查看結(jié)果iSum 和c均為5,調(diào)用正確。

              經(jīng)過以上幾個(gè)步驟的試驗(yàn),基本掌握了如何定義動(dòng)態(tài)庫函數(shù)以及如何在 C# 定義導(dǎo)入,有此基礎(chǔ),很快我實(shí)現(xiàn)了變長加密函數(shù)在 C# 中的調(diào)用,至此目標(biāo)實(shí)現(xiàn)。

              三、結(jié)論

              在 C# 中調(diào)用 C++ 編寫的動(dòng)態(tài)鏈接庫函數(shù),如果需要出口參數(shù)輸出,則需要使用指針,對(duì)于字符串,則需要使用雙重指針,對(duì)于 C# 的導(dǎo)入定義,則需要使用引用(ref)定義。

              對(duì)于函數(shù)返回值,C# 導(dǎo)入定義和 C++ 動(dòng)態(tài)庫函數(shù)聲明定義需要保持一致,否則會(huì)出現(xiàn)函數(shù)調(diào)用失敗。定義導(dǎo)入時(shí),一定注意 CharSet 和 CallingConvention 參數(shù),否則導(dǎo)致調(diào)用失敗或結(jié)果異常。運(yùn)行時(shí),動(dòng)態(tài)鏈接庫放在 C# 程序的目錄下即可,我這里是一個(gè) C# 的動(dòng)態(tài)鏈接庫,兩個(gè)動(dòng)態(tài)鏈接庫就在同一個(gè)目錄下運(yùn)行。


            ------------------------------------------------------------------------------------

            改寫c++ 到c#?(引)

            1. 在.h文件中找到所有的struct定義,先改成C#的。用Marshal.SizeOf得到其大小,初步進(jìn)行測試:看看它和在C/C++程序中用sizeof得到的大小是否相等
            2. 對(duì)應(yīng)所有C++的類,先寫出所有C#的類的原型,包括成員字段和函數(shù)。所有函數(shù)里先放著throw new NotImplementedException()這么一句,省得影響編譯
            3. 對(duì)應(yīng)于C++類的析構(gòu)函數(shù),在C#用Dispose Pattern實(shí)現(xiàn)
            4. 找到所有用到的Windows API和其他Native API,在C#用static extern方法聲明之,然后對(duì)不確定都否工作的那些(比如參數(shù)里頭有n個(gè)*的那種)進(jìn)行初步測試
            5. 在.h文件找到所有宏定義的常量,改成const的字段;而宏方法只能直接改成普通方法(C#沒有inline關(guān)鍵字)。對(duì)于用宏改了名字的類型,專門記載到一個(gè)文檔中,以后直接用文本替換(ctrl+h)搞定之
            6. 把.cpp文件的代碼逐步copy到C#代碼中,一個(gè)函數(shù)一個(gè)函數(shù)的改。基于C/C++標(biāo)準(zhǔn)庫和語言本身特性的代碼語句,一般都可在FCL和C#中找到對(duì)應(yīng)的做法,這也包括指針操作;至于API的調(diào)用,前面已經(jīng)準(zhǔn)備好了
            7. 編譯通過,測試一下

            posted on 2006-03-31 12:48 夢在天涯 閱讀(2107) 評(píng)論(5)  編輯 收藏 引用 所屬分類: CPlusPlusC#/.NET

            評(píng)論

            # re: C#中重用c/c++舊模塊 2006-03-31 13:21 沐楓

            你這項(xiàng)目要是沒有特殊要求,其實(shí)最簡單的方法是,采用C++/CLI來包裝原有的dll或代碼,然后就可以直接讓C#引用了。
            甚至可以直接用C++/CLI重編譯原有代碼,只增加一個(gè)或若干個(gè)接口類給.net引用。極端一點(diǎn)的,干脆不用C#,只用C++/CLI。
            這樣的做法是,簡單,快捷,并且完美。甚至可以讓代碼同時(shí)擁有本地代碼和托管代碼。從而為反編譯增加最大的難度。
            因?yàn)椴皇撬械慕涌诤蛃truct都能很方便的封裝成C#中去。  回復(fù)  更多評(píng)論   

            # re: C#中重用c/c++舊模塊 2006-03-31 15:10 Ninputer

            用C++/CLI重新編譯能夠成功才怪……用C++/CLI以PInvoke的方式調(diào)用輸出函數(shù)的非托管DLL幾率還大一些  回復(fù)  更多評(píng)論   

            # re: C#中重用c/c++舊模塊 2006-03-31 17:30 沐楓

            @Ninputer
            我試過不是這樣的。
            我這幾天曾把手頭上的幾個(gè)C++項(xiàng)目,一字不改,僅將編譯選項(xiàng)改為/clr,就編譯通過了。并且運(yùn)行效果完全正常。
            用reflector反編譯也成功的看到源碼(只是源碼有點(diǎn)不太好看)。
            這幾個(gè)項(xiàng)目有些是MFC項(xiàng)目,有些是API項(xiàng)目,都使用了模板和標(biāo)準(zhǔn)庫,以及第三方的DLL。  回復(fù)  更多評(píng)論   

            # re: C#中重用c/c++舊模塊 2006-04-02 17:16 sunmast

            COPY了我的內(nèi)容,但沒加我的鏈接,嗯 :-)  回復(fù)  更多評(píng)論   

            # re: C#中重用c/c++舊模塊 2006-04-03 08:39 夢在天涯

            @sunmast
            ^_^,不好意思啊 !可以留下你的連接嗎?  回復(fù)  更多評(píng)論   

            公告

            EMail:itech001#126.com

            導(dǎo)航

            統(tǒng)計(jì)

            • 隨筆 - 461
            • 文章 - 4
            • 評(píng)論 - 746
            • 引用 - 0

            常用鏈接

            隨筆分類

            隨筆檔案

            收藏夾

            Blogs

            c#(csharp)

            C++(cpp)

            Enlish

            Forums(bbs)

            My self

            Often go

            Useful Webs

            Xml/Uml/html

            搜索

            •  

            積分與排名

            • 積分 - 1805026
            • 排名 - 5

            最新評(píng)論

            閱讀排行榜

            久久WWW免费人成一看片| 久久天天躁狠狠躁夜夜2020老熟妇| 久久av高潮av无码av喷吹| 久久国产精品77777| 久久天天躁狠狠躁夜夜avapp| 人人狠狠综合久久88成人| 亚洲AV无码久久精品成人| 久久精品成人欧美大片| 久久久老熟女一区二区三区| 久久久久免费看成人影片| 亚洲中文字幕久久精品无码喷水| 久久久久成人精品无码中文字幕| 97久久超碰国产精品旧版| 久久精品9988| 亚洲午夜精品久久久久久浪潮 | 18岁日韩内射颜射午夜久久成人| 亚洲国产成人精品女人久久久 | 久久久久久亚洲精品不卡| 很黄很污的网站久久mimi色| 亚洲欧美一级久久精品| 综合久久一区二区三区| 精品国产青草久久久久福利| 97久久精品国产精品青草| 国产精品青草久久久久福利99 | 久久96国产精品久久久| 久久精品国产亚洲Aⅴ蜜臀色欲| 一本色道久久综合狠狠躁篇 | 久久久av波多野一区二区| 精品久久久久久中文字幕| 无码人妻久久一区二区三区蜜桃 | 亚洲女久久久噜噜噜熟女| 久久精品国产亚洲av高清漫画| 久久不见久久见免费影院www日本| 亚洲国产精品成人AV无码久久综合影院 | 久久久WWW成人| 久久久亚洲欧洲日产国码aⅴ| 久久se精品一区精品二区国产| 精品综合久久久久久888蜜芽| 狠狠精品干练久久久无码中文字幕| 亚洲中文久久精品无码ww16 | 伊人久久综合热线大杳蕉下载|