动态设置 DllImport 属性

9

我正在使用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版本进行构建。

2个回答

10

是的,这是可能的,您需要完成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 marshaller 帮助进行名称修饰。如果不确定导出名称是什么样子,请在 DLL 上使用 dumpbin.exe /exports。


确实,每个DLL API版本只有1个委托类型,再加上GetProcAddress,你就能摆脱这个问题了。 - Gregory Pakosz
这个问题在微软的博客文章中有详细介绍:http://blogs.msdn.com/b/jonathanswift/archive/2006/10/03/dynamically-calling-an-unmanaged-dll-from-.net-_2800_c_23002900_.aspx - Deanna

2

您无法更改dll的名称,但可以更改正在加载的库的路径(例如从注册表或配置文件中读取),并使用 LoadLibrary kernel32函数手动加载它:请参见我的答案


好的。但是在我的示例中,我已经指定了一个特定的函数原型,以便可以正确地封装参数,一些API函数具有复杂的结构作为参数。当以这种方式工作时,我该如何处理呢? - user226356
如果DLL的版本更改导致参数不同,那么我提到的方法就无法使用了。 - Gregory Pakosz

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接