首页 > 解决方案 > 为什么 LoadLibrary 失败而 DllImportAttribute 工作?

问题描述

我正在为使用其第三方系统之一执行 I/O 的客户端创建一个 .NET 应用程序。由于他们定期更改此系统的密码,我应该通过调用他们在专用目录中提供的本机 DLL 来动态检索它(除了我的 EXE 文件之外)。

但是,我无法使用LoadLibraryEx动态加载 DLL 。奇怪的是我可以使用DllImportAttribute调用库。

这是我到目前为止所做的:

根据this SO answer,我使用以下代码(在构造函数中)尝试动态加载DLL:

public PasswordProvider(string dllPath)
{
    if (!File.Exists(dllPath))
        throw new FileNotFoundException($"The DLL \"{dllPath}\" does not exist.");
    _dllHandle = NativeMethods.LoadLibraryEx(dllPath, IntPtr.Zero, LoadLibraryFlags.None);
    if (_dllHandle == IntPtr.Zero)
        throw CreateWin32Exception($"Could not load DLL from \"{dllPath}\".");

    var procedureHandle = NativeMethods.GetProcAddress(_dllHandle, GetPasswordEntryPoint);
    if (procedureHandle == IntPtr.Zero)
        throw CreateWin32Exception("Could not retrieve GetPassword function from DLL.");
    _getPassword = Marshal.GetDelegateForFunctionPointer<GetPasswordDelegate>(procedureHandle);
}
[DllImport(DllPath, EntryPoint = GetPasswordEntryPoint, CallingConvention = CallingConvention.StdCall, CharSet = CharSet.Unicode)]
private static extern long GetPassword(long systemId, string user, byte[] password);

这怎么可能?我认为背后的机制DllImportAttribute也在内部使用 LoadLibary。我的代码在哪里不同?我错过了一些明显的东西吗?

只是一些注释:

西蒙斯评论后编辑: NativeMethods定义如下:

private static class NativeMethods
{
    [DllImport("kernel32.dll", CharSet = CharSet.Auto, SetLastError = true)]
    public static extern IntPtr LoadLibrary(string dllName);

    [DllImport("kernel32.dll", CharSet = CharSet.Auto, SetLastError = true)]
    public static extern IntPtr LoadLibraryEx(string dllFileName, IntPtr reservedNull, LoadLibraryFlags flags);

    [DllImport("kernel32.dll", CharSet = CharSet.Auto, SetLastError = true)]
    public static extern IntPtr GetProcAddress(IntPtr moduleHandle, string procedureName);

    [DllImport("kernel32.dll")]
    public static extern bool FreeLibrary(IntPtr moduleHandle);
}

[Flags]
private enum LoadLibraryFlags : uint
{
    None = 0,
    DoNotResolveDllReferences = 0x00000001,
    LoadIgnoreCodeAuthorizationLevel = 0x00000010,
    LoadLibraryAsDatafile = 0x00000002,
    LoadLibraryAsDatafileExclusive = 0x00000040,
    LoadLibraryAsImageResource = 0x00000020,
    LoadLibrarySearchApplicationDir = 0x00000200,
    LoadLibrarySearchDefaultDirs = 0x00001000,
    LoadLibrarySearchDllLoadDir = 0x00000100,
    LoadLibrarySearchSystem32 = 0x00000800,
    LoadLibrarySearchUserDirs = 0x00000400,
    LoadWithAlteredSearchPath = 0x00000008
}

在 Hans Passant 发表评论后编辑:

总体目标是能够在我的应用程序(Windows 服务)运行时替换/更新本机 DLL。我检测到文件更改,然后重新加载 DLL。我不太确定在DllImportAttribute不重新启动服务的情况下这是否可行。

而且我应该更具体地说明实际问题:我无法使用 加载本机 DLL LoadLibraryEx,无论它是放在我的 EXE 旁边,还是在另一个随机文件夹中,还是在 SysWow64 中。为什么它适用于DllImportAttribute? 我很确定我的系统上不存在丢失的FastMM 子依赖 DLL(既不在实际 DLL 旁边,也不在任何 Windows 目录中)。

标签: c#.netdllinteroppinvoke

解决方案


@HansPassant 和 @David Heffernan 是对的:我实际上尝试加载两个不同版本的 DLL(其中一个具有 FastMM 子依赖项,一个没有)。感谢您的帮助,对于给您带来的不便,我们深表歉意。


推荐阅读