老规矩,在自己园子里发一份,也在小组里发一份。http://www.cnblogs.com/dwwwing
最近经常看到有人问托管非托管Dll调用的问题。对于动态库的调用其实很简单。网上很多代码都实现了Dll的静态调用方法。我主要谈论下动态库的动态加载。
对于托管动态库,实现动态加载很简单。
Code Assembly ass = 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 using 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 public delegate bool doDllFunction();//如果需要执行的函数有参数,可对之进行声明。 DllInvoke dllInvoke = new DllInvoke(filePath);//非托管dll文件路径 doDllFunction show = (doDllFunction) dllInvoke.Invoke(InitFunction,typeof(doDllFunction));// InitFunction为需要执行的函数名 show();//执行方法,可根据定义决定是否需要传参数
怎么样,是不是觉得很简单呢?哪么下面就自己动手尝试下吧。如果有疑问请联系我,我会为你详细解答。