C#如何調用一個非托管動態庫中的函數呢,比如用VC6寫的動態庫,總之C#調用動態庫的過程是比Java調用DLL動態庫方便快捷多了,下面舉例說明這個過程。
1。創建一個非托管動態庫
代碼如下:
//這一句是聲明動態庫輸出一個可供外不調用的函數原型.
extern "C" __declspec(dllexport) int add( int , int );
int add( int a, int b)
{
//實現這個函數returna+b;
}
注意上面代碼,一定要加上 extern"C" ,不能生成的動態庫中的導出函數名就不會是add,而是像?add@@YAHHH@Z樣子,後面只是通過函數名 add 來定位函數入口就會出問題。
保存成C或者CPP文件都可以,接下來就用命令 cl (這個命令VC6提供) 來編譯生成一個動態庫,命令如下:
C:>cl /LD MyLib.cpp
Microsoft (R) 32-bit C/C++ Optimizing Compiler Version 12.00.8804 for 80x86
Copyright (C) Microsoft Corp 1984-1998. All rights reserved.
MyLib.cpp
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
可以看到在C盤根目錄下生成了你要的動態庫 MyLib.Dll ,還伴隨著生成了MyLib.lib、MyLib.obj、MyLib.exp文件,上面命令cl的參數/LD就是生成動態庫文件
2.編寫C#程序調用動態庫
using System;
using System.Runtime.InteropServices; //這是用到DllImport時候要引入的包
public class InvokeDll{
[DllImport( "MyLib.dll" , CharSet=CharSet.Auto)]
staticexternint add( int a, int b); //聲明外部的標准動態庫, 跟Win32API是一樣.
public static void Main()
{
Console.WriteLine(add(10,30));
}
}
保存為InvokeDll.cs文件, 與MyLib.dll置於同一目錄, 編譯該文件.
C:>csc InvokeDll.cs
Microsoft (R) Visual C# .NET 編譯器版本 7.10.3052.4
用於 Microsoft (R) .NET Framework 版本 1.1.4322
版權所有 (C) Microsoft Corporation 2001-2002。保留所有權利。
將生成Invokedll.exe, 可以執行該文件.
C:>InvokeDll
40
我們看到C#調用了非托管動態庫的函數 add 。執行前保證 MyLib.dll 在能夠被 InvokeDll 程序加載到的路徑上。
回過頭來,如果我們在MyLib.cpp中沒有加上 extern"C" 在,那麼C中通過函數名 add 定位不到導出方法(因為函數名在動態庫中已經變了),執行invokeDll時就會出現如下錯誤。
C:>InvokeDll
未處理的異常: System.EntryPointNotFoundException: 無法在 DLL MyLib.dll 中找到名為 add 的入口點。
at InvokeDll.add(Int32 a, Int32 b)
at InvokeDll.Main()
對於沒有加上 extern "C" 的函數原型生成的動態庫,我們就得用別的方式來調用了,具體怎麼做,我現在還不知道。下面還有一個問題,上面的例子只是演示了動態庫中函數非常簡單的情況,如果函數傳遞的參數是指針,或者更復雜的數據類型,又如何操作呢?以後會深究的。