原文:刘武|c# 如何调用非托管函数
在目前的项目当中经常需要调用系统API,或者第三方的API,而这些API通常都不是基于.NET的,也就是所说的非托管函数,还好.NET为我们提供了平台调用服务,通过这个服务,就可以轻松的实现我们的需求。
调用过程其实比较简单,主要分以下几个步骤:
1) 找到函数的定义以及他所在的链接库(DLL文件)
以系统提供的BEEP函数为例(用指定的频率和时间发出蜂鸣声),他就是在动态链接库kernel32.dll中定义的。
在MSDN上可以找到他的函数签名为 BOOL Beep(DWORD dwFreq,DWORD dwDuration),可以看出该函数应该接受两个无符号的整数作为参数,第一个为频率,第二个为持续时间,并返回一个bool值。
2)在托管代码中创建函数原型
知道了函数的位置和签名,就可以为他编写托管定义了。看下面的代码
using System.Runtime.InteropServices;
[DllImport("kernel32.dll")]
static extern bool Beep(uint dwFreq, uint dwDuration);
首先引用了命名空间System.Runtime.InteropServices,他包含了所有平台调用必须的类,然后使用DllImport属性告诉编译器该函数的定义在kernel32.dll链接库中,由于该链接库位于system32目录中,因此我们不需要指定绝对路径。注意这里必须使用 static和extern修饰符,以声明该函数是外部实现的,同时函数的签名也应该和原来的保持一致。当然,我们也可以修改函数的名称,此时就必须指定入口点,如:
[DllImport("kernel32.dll", EntryPoint = "Beep")]
static extern bool Beep2(uint dwFreq, uint dwDuration);
EntryPoint就指明了原来函数的名称。
3)调用
自此我们就可以像使用托管代码一样调用该函数了 ,下面是完整的代码
using System.Runtime.InteropServices;
namespace ConsoleApplication1
{
class Program
{
static void Main(string[] args)
{
for (uint i = 100; i <= 20000; i++)
{
Beep(i, 5);
}
}
[DllImport("kernel32.dll")]
static extern bool Beep(uint dwFreq, uint dwDuration);
}
}
注:也可以把Beep函数放在一个单独的类里面,不过要声明为Public类型
当然,这只是一个最简单的例子,当涉及到数据类型的转换以及函数回调的时候,情况就会复杂多了,这个就姑且做为一个开篇吧。
附:
想查看系统API的定义,可参考MSDN Win32 and COM Development
想知道更多的API调用方法,请参考P/INVOKE
关于DllImport属性的更多说明,请参考MSDN