• <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++-----勇者無敵  
            流媒體、分布式實時系統(tǒng)、外掛、設計模式、C#
            日歷
            <2008年7月>
            293012345
            6789101112
            13141516171819
            20212223242526
            272829303112
            3456789
            統(tǒng)計
            • 隨筆 - 4
            • 文章 - 0
            • 評論 - 1
            • 引用 - 0

            導航

            常用鏈接

            留言簿(1)

            隨筆分類

            隨筆檔案(4)

            搜索

            •  

            最新評論

            閱讀排行榜

            評論排行榜

             
            被C#調(diào)用的DLL一般只需要把導出的函數(shù)以適當?shù)男问匠尸F(xiàn)即可調(diào)用,比如
            extern "C" __declspec(dllexport)
            BOOL Integrate3 (){...},這樣的函數(shù),在C#里面聲明如:

            [DllImport("xxx.dll", CallingConvention = CallingConvention.StdCall, CharSet = CharSet.Unicode)]
                    public static extern bool Integrate3();,這里的調(diào)用相對是簡單的,而有些數(shù)據(jù)類型則必須通過MarshalAs來做托管類型的轉(zhuǎn)換,如:

            extern "C" __declspec(dllexport)
            BOOL Integrate (LPCWSTR file1, LPCWSTR file2, LPCWSTR outputFile){...}

            由于數(shù)據(jù)類型不一致,所以在聲明時要注意把類型轉(zhuǎn)換過來。

            [DllImport("xxx.dll", CallingConvention = CallingConvention.StdCall, CharSet = CharSet.Unicode)]
                    public static extern bool Integrate([In, MarshalAs(UnmanagedType.LPWStr)]string file1,
                        [In, MarshalAs(UnmanagedType.LPWStr)]string file2, [In, MarshalAs(UnmanagedType.LPWStr)]string outputFile);

            這樣調(diào)用基本是沒有問題,重點在于數(shù)據(jù)類型的轉(zhuǎn)換。多試過幾次了就不問題了。

            另外一個小小的實踐經(jīng)驗就是在C#中調(diào)試C++的DLL,知道了就是一句話,不知道就要搞半天,在C#項目屬性中“啟用調(diào)試項”中一項:“啟用非托管代碼調(diào)試”,鉤上這個,就萬事大吉了,就像你調(diào)試一般的程序一樣。

            C#調(diào)用C++的DLL時,參數(shù)傳遞便成了一個問題。今天我碰到的一個問題是,在C++中導出的函數(shù)的參數(shù)是string類型的,在C#中通過string的參數(shù)調(diào)用時,便會出現(xiàn)該內(nèi)存已損壞或不能讀取的異常信息。后來我把C++的導出函數(shù)的參數(shù)由string改為LPTSTR類型,也即char*類型,然后在C#中對應的參數(shù)改為StringBuilder類型,既解決了傳進去的參數(shù)問題,又解決了傳出參數(shù)的問題。

            DllImport知識拓展:DllImport 
             .net 框架程序可以通過靜態(tài) DLL 入口點的方式來訪問本機代碼庫。DllImport 屬性用于指定包含外部方法的實現(xiàn)的dll 位置。

            DllImport 屬性定義如下:
              
            namespace System.Runtime.InteropServices
            {
             [AttributeUsage(AttributeTargets.Method)]
             public class DllImportAttribute: System.Attribute
             {
              public DllImportAttribute(string dllName) {...}
              public CallingConvention CallingConvention;
              public CharSet CharSet;
              public string EntryPoint;
              public bool ExactSpelling;
              public bool PreserveSig;
              public bool SetLastError;
              public string Value { get {...} }
             }
            }
              
              說明:
              
             1、DllImport只能放置在方法聲明上。 
                2、DllImport具有單個定位參數(shù):指定包含被導入方法的 dll 名稱的 dllName 參數(shù)。 
                3、DllImport具有五個命名參數(shù):
                 a、CallingConvention 參數(shù)指示入口點的調(diào)用約定。如果未指定 CallingConvention,則使用默認值 CallingConvention.Winapi。
                 b、CharSet 參數(shù)指示用在入口點中的字符集。如果未指定 CharSet,則使用默認值 CharSet.Auto。
               c、EntryPoint 參數(shù)給出 dll 中入口點的名稱。如果未指定 EntryPoint,則使用方法本身的名稱。
               d、ExactSpelling 參數(shù)指示 EntryPoint 是否必須與指示的入口點的拼寫完全匹配。如果未指定 ExactSpelling,則使用默認值 false。
               e、PreserveSig 參數(shù)指示方法的簽名應當被保留還是被轉(zhuǎn)換。當簽名被轉(zhuǎn)換時,它被轉(zhuǎn)換為一個具有 HRESULT 返回值和該返回值的一個名為 retval 的附加輸出參數(shù)的簽名。如果未指定    PreserveSig,則使用默認值 true。
                  f、SetLastError 參數(shù)指示方法是否保留 Win32"上一錯誤"。如果未指定 SetLastError,則使用默認值 false。 
             4、它是一次性屬性類。
             5、此外,用 DllImport 屬性修飾的方法必須具有 extern 修飾符。

            下面是 C# 調(diào)用 Win32 MessageBox 函數(shù)的示例:
              
            using System;
            using System.Runtime.InteropServices;
            class MainApp
            {

                 //通過DllImport引用user32.dll類。MessageBox來自于user32.dll類
             [DllImport("user32.dll", EntryPoint="MessageBox")]
             public static extern int MessageBox(int hWnd, String strMessage, String strCaption, uint uiType);
             public static void Main()
             {
              MessageBox( 0, "您好,這是 PInvoke!", ".net", 0 );
             }
            }
            面向?qū)ο蟮木幊陶Z言幾乎都用到了抽象類這一概念,抽象類為實現(xiàn)抽象事物提供了更大的靈活性。C#也不例外, C#通過覆蓋虛接口的技術(shù)深化了抽象類的應用。欲了解這方面的知識,請看下一節(jié)-覆蓋虛接口

            這里講述的是C#調(diào)用標準動態(tài)庫的問題, 在我以前的文件中講到過, C#調(diào)用Win32API, 原理是一樣的. 這里我詳細講解用C寫一個標準的動態(tài)庫, 然后讓C#調(diào)用. (本篇適合初學者, 中間沒有任何冗余代碼, 簡潔明了)
            軟件環(huán)境: VC6.0(當然其他版本的VC5也可以)
            1.制作標準動態(tài)庫


            __declspec(dllexport) int __cdecl add(int, int);//這一句是聲明動態(tài)庫輸出一個可供外不調(diào)用的函數(shù)原型.
            int add(int a,int b) {//實現(xiàn)這個函數(shù)
            return a+b;
            }


            以上簡單3行代碼,聲明一個add的方法, 輸入?yún)?shù)是兩個int參數(shù),返回這兩個數(shù)之和. 保存為MyLib.c
            然后執(zhí)行編譯命令.
            H:\XSchool\C#-School\HowTo>cl /LD MyLib.c
            Microsoft (R) 32-bit C/C++ Optimizing Compiler Version 12.00.8168 for 80x86
            Copyright (C) Microsoft Corp 1984-1998. All rights reserved.

            MyLib.c
            Microsoft (R) Incremental Linker Version 6.00.8447
            Copyright (C) Microsoft Corp 1992-1998. All rights reserved.

            /out:MyLib.dll
            /dll
            /implib:MyLib.lib
            MyLib.obj
               Creating library MyLib.lib and object MyLib.exp

            確信有以上輸出, 說明編譯成功生成了動態(tài)庫.

            2.編寫C-Sharp程序調(diào)用該動態(tài)庫
            using System;
            using System.Runtime.InteropServices;//這是用到DllImport時候要引入的包

            public class InvokeDll {
            [DllImport("MyLib.dll", CharSet=CharSet.Auto)]
            static extern int add(int a,int b);//聲明外部的標準動態(tài)庫, 跟Win32API是一樣的.

            public static void Main() {
            Console.WriteLine(add(10,30));
            }
            }
            保存為InvokeDll.cs文件, 與MyLib.dll置于同一目錄, 編譯該文件.
            H:\XSchool\C#-School\HowTo>csc invokedll.cs
            將生成Invokedll.exe, 可以執(zhí)行該文件.
            以上是C-Sharp調(diào)用標準動態(tài)庫的全過程, 本來覺得很簡單的東西, 一直都沒有想寫, 碰巧今日遇一朋友問及此事, 就順便寫了下來.

            posted on 2008-07-30 16:54 李明坤 閱讀(6293) 評論(1)  編輯 收藏 引用
            評論:
            • # re: C#中調(diào)試C++的DLL[未登錄]  haha Posted @ 2015-06-24 11:53
              你好,我也遇到C#調(diào)用C++DLL的情況,但是有所不同的是,我引用的這個C++DLL又引用了另外一個C++DLL,我可以調(diào)試跟到第一層C++DLL,卻無法調(diào)試跟到第二個C++DLL,不知這個問題怎么解決,求教一下  回復  更多評論   

             
            Copyright © 李明坤 Powered by: 博客園 模板提供:滬江博客
            久久精品国产72国产精福利| 日韩人妻无码精品久久免费一 | 日韩久久久久久中文人妻| 无码超乳爆乳中文字幕久久 | 一本大道久久a久久精品综合| 欧美性大战久久久久久| 久久99精品国产一区二区三区| 国产精品美女久久久久| 久久香综合精品久久伊人| 久久精品国产亚洲av日韩| 久久久SS麻豆欧美国产日韩| 精品久久久久国产免费| 久久精品成人免费看| 国产午夜精品久久久久九九电影| 狠狠狠色丁香婷婷综合久久俺| 久久综合狠狠综合久久综合88| 精品久久久久久无码不卡| 久久久久亚洲av成人网人人软件| 久久夜色精品国产www| 久久久久国色AV免费观看| 亚洲国产成人精品女人久久久| 久久久久久免费视频| 狠狠色婷婷综合天天久久丁香| 999久久久免费国产精品播放| 亚洲精品白浆高清久久久久久| 日韩欧美亚洲综合久久| 久久久久久狠狠丁香| 久久香综合精品久久伊人| 久久精品国产99久久无毒不卡| 99热精品久久只有精品| 漂亮人妻被中出中文字幕久久| 精品国产一区二区三区久久久狼| 久久无码一区二区三区少妇| 午夜精品久久久久久久| 日韩欧美亚洲国产精品字幕久久久| 久久亚洲AV成人出白浆无码国产| 国产一区二区精品久久凹凸| 国产一级持黄大片99久久| 亚洲国产精久久久久久久| 久久本道久久综合伊人| 99久久人妻无码精品系列|