1

要求:Qt 小部件在 Qt 共享库加载时显示,对于非 Qt应用程序。

经过一些网络搜索,我发现:

  1. 所有 Qt 小部件都必须存在于“主线程”中,“主线程”是 Qt 对象创建的第一个线程。因此,创建一个非 Qt 线程(std::thread),然后在该线程中创建 QApplication 和其他一些小部件应该可以工作,但不能。

  2. 在创建 QApplication 之前,不要在该非 Qt 线程中创建任何 Qt 相关对象或调用任何 Qt 相关静态方法。

  3. 线程解决方案不适用于 Mac OS,我的目标平台仅为 Windows,所以没关系。

  4. 就我而言,如果应用程序加载我的 Qt 库,并调用显示小部件的方法,它就可以工作。但由于某种原因,调用者无法手动调用我的 lib 方法。

  5. 如果宿主应用程序(加载共享库的应用程序)是 Qt 应用程序,您应该调用 QApplication::processEvents(),而不是 QApplication::exec()。就我而言,我应该在该线程中调用 QApplication::exec() 。

源代码在这里:

  • dll主要版本
BOOL APIENTRY DllMain(HMODULE hModule,
                      DWORD ul_reason_for_call,
                      LPVOID lpReserved)
{
    if (ul_reason_for_call == DLL_PROCESS_ATTACH) {
        auto t = std::thread([]() {
            // setCodecForLocale is in the same thread, 
            // call it before QApplication created should be OK.

            QTextCodec::setCodecForLocale(QTextCodec::codecForName("GBK"));
            int i = 0;
            int argc = 0;
            QApplication app(argc, 0);

            auto dialogLogin = new DialogLogin(); // custom widget

            dialogLogin->setModal(true);
            dialogLogin->show();
            app.exec(); // app.processEvents() not work, too.
        });

        t.join(); // wait for thread ends in dllMain should be BAD, test only
    }

    return true;
}
  • 简单 C++ 静态类版本
class LibExecutor {
public:
    LibExecutor()
    {
        auto t = std::thread([]() {
            QTextCodec::setCodecForLocale(QTextCodec::codecForName("GBK"));
            int argc = 0;
            QApplication app(argc, 0);

            auto dialogLogin = new DialogLogin();
            dialogLogin->setModal(true);
            dialogLogin->show();
            app.exec();
        });

        t.join();
    }
};

static LibExecutor libExecutor;

两个版本都成功调用小部件初始化内容,但小部件未显示。

这是我使用 Qt 加载 lib 进行测试的方法,但是,我使用 Win32 API 加载 lib 的事件也失败了。

#include "mainwindow.h"
#include <QApplication>
#include <QLibrary>

int main(int argc, char* argv[])
{
    QLibrary lib("F:/lib_location/lib_name.dll");

    if (lib.load()) {
        qDebug() << "load ok!";
    } else {
        qDebug() << "load error!";
    }
}
4

1 回答 1

0

这是一个工作示例。使用 Qt 5.12 和 MSVC2017 和 MinGW 进行测试。

// main.cpp
int main(int argc, char *argv[])
{
    run_mylib_t *f= nullptr;
    HMODULE lib = LoadLibraryA("..\\mylib\\debug\\mylib.dll");

    if (!lib) {
        qDebug() << "Failed to load library;";
        return -1;
    }

    f = reinterpret_cast<run_mylib_t *>(GetProcAddress(lib, "run_mylib"));

    if (!f) {
        qDebug() << "Failed to get function";
        return -1;
    }

    f(argc, argv);      

    return 0;
}

// mylib.h
extern "C" MYLIBSHARED_EXPORT int run_mylib(int argc, char *argv[]);
using run_mylib_t = int(int, char *[]);

// mylib.cpp
int loop(int argc, char *argv[])
{
    QApplication a(argc, argv);
    MainWindow w;
    w.show();

    return a.exec();
}

int run_mylib(int argc, char *argv[])
{
    auto lambda = [argc, argv]() {loop(argc, argv); };

    std::thread thread(lambda);
    thread.join();

    return 0;
}

注意,如果在创建线程之前使用Qt函数,Qt会检测到它不在主线程中,进程会崩溃。这就是为什么我不使用QLibrary.

Qt 不支持此用例。所以如果你现在让它工作,你不能保证它将来会工作。

您不能像这样同时加载 2 个 dll。

根据您在主应用程序中执行的操作,可能会发生某些 Qt 功能无法按预期工作的情况。例如,Qt 可能会期望来自 Windows 的消息,但永远不会得到它们,因为它们将由真正的主线程处理。

关于 DllMain

从 Windows 文档:

警告

在 DLL 入口点中可以安全地执行的操作有很大的限制。有关在 DllMain 中调用不安全的特定 Windows API,请参阅一般最佳实践。如果您需要除了最简单的初始化之外的任何其他内容,请在 DLL 的初始化函数中执行此操作。您可以要求应用程序在 DllMain 运行之后和调用 DLL 中的任何其他函数之前调用初始化函数。

-- https://docs.microsoft.com/en-us/windows/desktop/dlls/dllmain

和动态链接库最佳实践:

永远不要在 DllMain 中执行以下任务:

  • 调用 LoadLibrary 或 LoadLibraryEx(直接或间接)。这可能会导致死锁或崩溃。
  • 调用 GetStringTypeA、GetStringTypeEx 或 GetStringTypeW(直接或间接)。这可能会导致死锁或崩溃。
  • 与其他线程同步。这可能会导致死锁。
  • 获取由等待获取加载程序锁的代码拥有的同步对象。这可能会导致死锁。
  • 使用 CoInitializeEx 初始化 COM 线程。在某些条件下,该函数可以调用 LoadLibraryEx。
  • 调用注册表函数。这些函数在 Advapi32.dll 中实现。如果 Advapi32.dll 未在您的 DLL 之前初始化,则 DLL 可以访问未初始化的内存并导致进程崩溃。
  • 调用 CreateProcess。创建一个进程可以加载另一个 DLL。
  • 调用 ExitThread。在 DLL 分离期间退出线程可能会导致再次获取加载程序锁,从而导致死锁或崩溃。
  • 调用 CreateThread。如果你不与其他线程同步,创建一个线程可以工作,但它是有风险的。
  • 创建命名管道或其他命名对象(仅限 Windows 2000)。在 Windows 2000 中,命名对象由终端服务 DLL 提供。如果此 DLL 未初始化,对 DLL 的调用可能会导致进程崩溃。
  • 使用动态 C 运行时 (CRT) 中的内存管理功能。如果 CRT DLL 未初始化,对这些函数的调用可能会导致进程崩溃。
  • 调用 User32.dll 或 Gdi32.dll 中的函数。某些函数加载另一个 DLL,该 DLL 可能未初始化。
  • 使用托管代码。

-- https://docs.microsoft.com/en-us/windows/desktop/dlls/dynamic-link-library-best-practices

由此我可以告诉您,您将无法创建QApplication和运行 Qt 应用程序DllMain,至少有以下原因:

  • Qt 将加载插件(至少qwindows.dll)使用LoadLibrary. 如果您使用任何音频或图像或 sql 数据库,Qt 也会尝试加载相应的插件(例如qjpeg.dll)。
  • Qt 也可能会尝试访问注册表,特别是如果您使用QSettings本机格式。
  • Qt 可以创建线程。特别是如果您使用网络或 Qt Quick。
  • mallocQt 将使用诸如or之类的内存管理功能free
于 2019-02-22T07:44:03.793 回答