QT中的延迟DLL加载

3

QT中是否可以进行延迟DLL加载?例如,我的程序依赖于某个第三方DLL,我想要将其移除。

在.pro文件中应该写什么来移除依赖项?

我可以将DLL存储在资源中吗?

我可以“全局”加载DLL吗?这样,DLL中的某些函数(例如func1)将保持在我的代码中作为func1。

2个回答

6

是的,您需要使用QLibrary类。它专门用于在运行时加载共享库。

在这种情况下,您不需要在.pro文件中添加任何内容。但是,您需要确保DLL在目标计算机上可用。它可以处于系统可以找到它的位置(Windows上的PATH,Linux上的LD_LIBRARY_PATH,Mac上的DYLD_LIBRARY_PATH),或者您可以为应用程序提供绝对路径。

如果查看文档,您将了解如何将库函数映射到代码中的函数指针,以便"func1()"将指向库中的函数。

[编辑]

这里有一个从资源生成的临时文件中加载dll的概念验证。在我看来,这是一个非常糟糕的想法,但至少它可以在我的Windows 7机器上工作。如果您想做类似的事情,您将不得不处理清理临时文件,检查重复等。

foo.cpp,编译为共享库

#include <QtCore/qglobal.h>

extern "C" Q_DECL_EXPORT int foo(int value) {
  return value + 42;
}

bar.pro,注意没有引用foo库。

SOURCES += main.cpp
RESOURCES += resources.qrc

main.cpp

#include <QtCore>
#include <iostream>

int main(int argc, char **argv) {
  QCoreApplication app(argc, argv);

  // Copy resource dll to temporary file.
  QFile::copy(":/lib/Foo.dll", QDir::temp().filePath("Foo.dll"));

  // Load the temporary file as a shared library.
  QLibrary foo_lib(QDir::temp().filePath("Foo.dll"));
  typedef int (*FooDelegate)(int);
  FooDelegate foo = (FooDelegate)foo_lib.resolve("foo");

  if (foo) {
    std::cout << "foo(13) = " << foo(13) << std::endl;
  }
}

+1。感谢您的回答。但我的问题是DLL在目标计算机上不可用...所以我想消除依赖性,将其存储在资源中,并在可能的情况下动态加载。此外,用户应该只看到一个.exe文件,而无需安装... - Eddie
哦,我明白了。你可能已经想到了这一点,但是你能否获得一个可以静态链接的库版本呢?这听起来像是你最终想要实现的目标。 - Dave Mateer
1
如果您无法进行静态链接,可以尝试在main()方法中将DLL作为二进制资源加载并放置在某个特定位置,例如系统的临时目录,然后使用QLibrary从该位置加载它。但这是一种非常糟糕的代码味道,可能会导致安全问题。 - Dave Mateer
1
我认为你必须创建临时文件——这是系统库加载函数所期望的。由于所有加载都是基于魔术字符串在运行时进行的,因此没有依赖性;如果出现问题,你只会遇到一个令人不快的崩溃。 - Dave Mateer
谢谢您的帮助,我认为您的回答解答了我的问题。 - Eddie
显示剩余2条评论

3
我只能从Microsoft编译器的角度来说...... 这更少取决于QT想要什么,而更多地取决于何时何地访问延迟加载的代码和/或数据。
一般来说,必须使用/delayload:[dllname]选项告诉链接器该DLL是延迟加载的,并与delayimp.lib进行链接。 您可以在.cpp文件中添加库,例如: #pragma comment(lib, "delayimp") 这将消除在链接器命令行中指定它的需要。
我不确定在.pro文件中添加延迟加载开关以完成链接器命令所需的步骤。
请参阅此链接以获取Microsoft文档。

谢谢您的回答。但是我不知道在QT中应该在哪里写/delayload:[dllname],而且QT会返回“忽略#pragma comment”。 - Eddie

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