原文:
托管非托管Dll动态调用
最近经常看到有人问托管非托管Dll调用的问题。对于动态库的调用其实很简单。网上很多代码都实现了Dll的静态调用方法。我主要谈论下动态库的动态加载。
对于托管动态库,实现动态加载很简单。
Code= Assembly.LoadFile(filePath);//这里是动态库的路径。
Type tp = ass.GetType(dllType);//dllType是你所需要调用的动态库文件的命名空间+类名(NameSpace.Class)
MethodInfo method = tp.GetMethod(function);//需要执行的函数
object ob = Activator.CreateInstance(tp);//创建对象
method.Invoke(ob, null);//执行函数,后一个参数即为执行函数需要的参数,若无则为null。
对于非托管dll的调用。相对托管动态库来说麻烦一点,但是也是很简单的。
使用三个API函数:LoadLibrary,GetProcAddress,FreeLibrary。
使用LoadLibrary将非托管Dll加载到内存中。调用GetProcAddress获取需调用的函数指针。将非托管函数指针转换为委托即可。最后调用FreeLibrary释放加载加载的非托管内存(加载后必须释放非托管内存)。
Code System;
using System.Collections.Generic;
using System.Text;
using System.Runtime.InteropServices;
namespace DllTest
{
public class DllInvoke
{
[DllImport("Kernel32.dll")]
private static extern IntPtr LoadLibrary(string path);
[DllImport("Kernel32.dll")]
private static extern IntPtr GetProcAddress(IntPtr lib,string FunName);
[DllImport("kernel32.dll")]
public static extern bool FreeLibrary(IntPtr lib);
private IntPtr libr;
public DllInvoke(string path)
{
libr = LoadLibrary(path);
}
public Delegate Invoke(string funName, Type type)
{
IntPtr api = GetProcAddress(libr, funName);
return (Delegate)Marshal.GetDelegateForFunctionPointer(api, type);
}
~DllInvoke()
{
FreeLibrary(libr);//释放。必须的
}
}
}
完成上面的函数声明后,接着我们先定一个委托.
Code delegate bool doDllFunction();//如果需要执行的函数有参数,可对之进行声明。
DllInvoke dllInvoke = new DllInvoke(filePath);//非托管dll文件路径
doDllFunction show = (doDllFunction) dllInvoke.Invoke(InitFunction,typeof(doDllFunction));// InitFunction为需要执行的函数名
show();//执行方法,可根据定义决定是否需要传参数