我正在使用PInvoke和DllImport属性来使用外部非托管dll。例如。
[DllImport("mcs_apiD.dll", CharSet = CharSet.Auto)]
private static extern byte start_api(byte pid, byte stat, byte dbg, byte ka);
我想知道是否有可能以某种方式动态地更改dll文件详细信息(在此示例中为mcs_apiD.dll),例如,如果我想针对另一个dll版本构建
答案 0 :(得分:7)
是的,这是可能的,你必须完成P / Invoke marshaller所做的部分工作。加载DLL并查找导出函数的入口点。首先声明一个签名与导出函数匹配的委托:
private delegate byte start_api(byte pid, byte stat, byte dbg, byte ka);
然后使用这样的代码:
using System.ComponentModel;
using System.Runtime.InteropServices;
...
static IntPtr dllHandle;
...
if (dllHandle == IntPtr.Zero) {
dllHandle = LoadLibrary("mcs_apiD.dll");
if (dllHandle == IntPtr.Zero) throw new Win32Exception();
}
IntPtr addr = GetProcAddress(dllHandle, "_start_api@16");
if (addr == IntPtr.Zero) throw new Win32Exception();
var func = (start_api)Marshal.GetDelegateForFunctionPointer(addr, typeof(start_api));
var retval = func(1, 2, 3, 4);
...
[DllImport("kernel32.dll", CharSet = CharSet.Auto, SetLastError = true)]
private static extern IntPtr LoadLibrary(string name);
[DllImport("kernel32.dll", CharSet = CharSet.Ansi, SetLastError = true)]
private static extern IntPtr GetProcAddress(IntPtr hModule, string name);
当然,有很多方法可以解决这个问题。请注意,您必须使用DLL中的实际导出名称,您不再从P / Invoke编组器获得帮助以进行名称修饰。如果您不确定导出名称是什么,请在DLL上使用dumpbin.exe / exports。
答案 1 :(得分:2)
您无法更改dll的名称,但您可以更改正在加载的库的路径(例如通过从注册表或配置文件中读取它)并使用LoadLibrary
kernel32的函数手动加载它: see my answer there