C# 动态设置DllImport属性

C# 动态设置DllImport属性,c#,pinvoke,C#,Pinvoke,我使用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版本构建,您不能更改dll的名称,但可以更

我使用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版本构建,您不能更改dll的名称,但可以更改正在加载的库的路径(如从注册表或配置文件中读取)并使用
LoadLibrary
kernel32的函数手动加载它。

是的,这是可能的,您必须完成p/Invoke封送器所做的部分工作。加载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。

Ok。但在我的示例中,我指定了一个特定的函数原型,以便正确地标记参数,一些api函数具有复杂的结构作为参数。如果参数从DLL的一个版本更改为另一个版本,那么我提到的方法就真的不走运了,每个版本的DLL API加上GetProcAddress都有一个委托类型,你会从中解脱出来的。这是一篇微软的博客文章,网址是:2800_c_23002900.aspx