浅浅的说说DllImport属性

时间:2022-07-13 18:09:16

    DllImport属性的功能:

    [MSDN]从托管应用程序调用非托管代码。(托管代码指的是必须依靠.NET框架解释运行的代码,非托管代码一般指的是传统的不需要借助.NET框架解释的代码。在.NET出现之前,如VB,C++,DELPHI编写的程序都是非托管代码。)

     

    在开发时会经常遇到的,比如商务拨号系统中,直接和话媒硬件打交道的代码集成在一个用C编写的dll中,而在.net下引用这个dll中的方法时,就需要用到DllImport。

     

    比如

[DllImport("phonic_ubox.dll")]

public static extern void ubox_close(); // extern修饰符用于声明在外部实现的方法

 

DLL对应的API文档

void WINAPI    ubox_close(void)

功  能:关闭设备

说  明:  最后程序退出,必须调用这个函数,否则会出现异常

 

示例

使用 C 程序创建一个 DLL,使用 C# 程序调用该 DLL。

 

    int __declspec(dllexport) SampleMethod(int i)  
{
return i*10;
}

从 C# 程序内调用。

    using System;  
using System.Runtime.InteropServices;
public class MainClass
{
[DllImport("Cmdll.dll")]
public static extern int SampleMethod(int x);

static void Main()
{
Console.WriteLine("SampleMethod() returns {0}.", SampleMethod(5));
}
}

 

输出

SampleMethod() returns 50.

 

DllImport会按照顺序自动去寻找引用的dll:

 1、项目下的bin目录下 2、System32目录 3、环境变量目录。即只需把ddl放入任一目录下即可。

 

引入问题:非托管dll放在bin下无济于事。

Asp.Net Team的官方解决方案如下:托管的很好办,直接被使用的需要引用,间接使用的需要拷贝到bin目录下。非托管的处理会比较麻烦.实际上,你拷贝到bin没有任何帮助,因为CLR会把文件拷贝到一个临时目录下,然后在那运行web,而CLR 只会拷贝托管文件,这就是为什么我们明明把非托管的dll放在了bin下却依然提示不能加载模块了. 

 

对于自己部署的应用程序,引用非托管dll完全采用后两种方式。

 

然而,如果我们用的是虚拟空间,我们是没办法把注 册PATH变量或者把我们自己的DLL拷到system32目录的。

 

网上找到的解决方法,因为尚未涉及到,所以只是简单的罗列下:

DllImport里面只能用字符 串常量,而不能够用Server.MapPath(@"~/Bin/Judge.dll")来确定物理路径。经过一翻研究,终于想到了一个完美的解决办法首先我们用

 

[DllImport("kernel32.dll")]

privateexternstatic IntPtr LoadLibrary(String path);

 

[DllImport("kernel32.dll")]

privateexternstatic IntPtr GetProcAddress(IntPtr lib, String funcName);

 

[DllImport("kernel32.dll")]

privateexternstaticbool FreeLibrary(IntPtr lib);

分别取得了LoadLibrary和GetProcAddress函数的地址,再通过这两个函数来取得我们的DLL里面的函数。

我们可以先用Server.MapPath(@"~/Bin/Judge.dll")来取得我们的DLL的物理路径,然后再用LoadLibrary进行载入,最后用GetProcAddress取得要用的函数地址

 

以下自定义类的代码完成LoadLibrary的装载和函数调用:

 

publicclass DllInvoke

{           

[DllImport("kernel32.dll")]

privateexternstatic IntPtr LoadLibrary(String path);

 

[DllImport("kernel32.dll")]  

privateexternstatic IntPtr GetProcAddress(IntPtr lib, String funcName);

 

[DllImport("kernel32.dll")]    

privateexternstaticbool FreeLibrary(IntPtr lib);    

 

private IntPtr hLib;  

 

public DllInvoke(String DLLPath)  

{          

hLib = LoadLibrary(DLLPath); 

}      

 

~DllInvoke()    

{       

FreeLibrary(hLib); 

}       

 

//将要执行的函数转换为委托 

public Delegate Invoke(String APIName,Type t)    

{          

IntPtr api = GetProcAddress(hLib, APIName);  

return (Delegate)Marshal.GetDelegateForFunctionPointer(api,t);    

}

}

下面代码进行调用

 

publicdelegateint Compile(String command, StringBuilder inf);

//编译

DllInvoke dll = new DllInvoke(Server.MapPath(@"~/Bin/Judge.dll"));

Compile compile = (Compile)dll.Invoke("Compile",typeof(Compile));

StringBuilder inf;

compile(@gcc a.c -o a.exe“,inf);//这里就是调用我的DLL里定义的Compile函数



小结

我说DllImport的伟大之处在于,.net平台没有闭关锁国,能够容纳别人,集百家之长。做人也该如此。