• <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: 博客園 模板提供:滬江博客
            亚洲香蕉网久久综合影视| 国产精品禁18久久久夂久| 日批日出水久久亚洲精品tv| 伊人久久亚洲综合影院| 久久综合狠狠综合久久| 国产精品美女久久久免费| 亚洲AV日韩AV永久无码久久| 久久精品国产99久久香蕉| 伊人久久精品无码av一区| 久久91这里精品国产2020| 亚洲中文字幕无码一久久区| 国内精品久久久久久久涩爱 | 国内精品伊人久久久久AV影院| 欧美综合天天夜夜久久| 国产美女亚洲精品久久久综合| 久久中文字幕一区二区| 欧洲人妻丰满av无码久久不卡| 久久精品成人免费观看97| 久久91精品国产91久久小草 | 99久久国产热无码精品免费| 久久综合色之久久综合| 91久久香蕉国产熟女线看| 日韩久久久久久中文人妻| 欧美亚洲国产精品久久高清 | 性高朝久久久久久久久久| 99久久免费国产精品热| 97久久国产综合精品女不卡| 久久影视综合亚洲| 久久久久99精品成人片三人毛片 | 久久久精品一区二区三区| 久久久久高潮毛片免费全部播放| 久久久国产视频| 久久久精品久久久久影院| 久久久国产一区二区三区| 国产精品成人精品久久久| 国产精品无码久久久久| 国产成人久久精品麻豆一区| 国产成人精品久久亚洲| 久久国产精品偷99| 久久综合给合综合久久| 色99久久久久高潮综合影院 |