Qt线程中启动主事件循环

在没有main()入口且不希望阻塞主线程的情况下,如何在Qt中使用信号槽及QTimer等类?本文通过实例展示了如何在Windows SDK的dll中,利用std::thread在其他线程中启动QCoreApplication的主事件循环,详细解析了启动过程及退出事件循环的注意事项。

场景举例:

  • 没有main()启动主线程,但希望用Qt的类和信号槽
  • 不希望app.exec()阻塞主线程

比如我最近在制作一个Windows SDK,封装成dll,只暴露两个简单全局函数作为接口:

extern "C" __declspec(dllexport) int start(const char*, const char*);
extern "C" __declspec(dllexport) void stop();

没有主事件循环,我就无法使用信号槽机制,我甚至无法正常使用QTimer、QProcess等等

那如何启动主事件循环呢?

QCoreApplication app(argc, argv);

...

app.exec();

但我没有main()入口,怎么办?

线程中启动主事件循环就可以——主事件循环不一定要在主线程中启动哦

注意不能使用QThread,因为QThread的正常使用也依赖事件循环

使用std::thread


我的实现:

1、定义全局对象

static MyObject* m = nullptr;

2、定义启动参数对象

struct InputArgs{
  int argc;
  char **argv;
};

3、获取启动参数——调用程序的可执行文件路径

WCHAR exePath[MAX_PATH+1];
String exeFullPath;
DWORD len = GetModuleFIleNameW(nullptr, exePath, MAX_PATH);    // #include 
Qt中,不同场景下启动事件循环的方式有所不同。 ### 主线程事件循环线程事件循环由`QApplication::exec()`启动,`QApplication`调用的是`QGuiApplication`的`exec()`,而`QGuiApplication`内部调用的是`QCoreApplication`的`exec()`。示例代码如下: ```cpp #include <QApplication> #include <QWidget> int main(int argc, char *argv[]) { QApplication a(argc, argv); QWidget w; w.show(); // 启动线程事件循环 return a.exec(); } ``` ### 子线程事件循环线程通过`QThread::exec()`启动事件循环。可以通过子类化`QThread`并重写`run()`方法,在`run()`方法中调用`exec()`来开启子线程事件循环。示例代码如下: ```cpp #include <QThread> #include <QDebug> class MyThread : public QThread { Q_OBJECT protected: void run() override { qDebug() << "Thread started"; // 启动线程事件循环 exec(); qDebug() << "Thread finished"; } }; ``` 使用时: ```cpp #include <QCoreApplication> int main(int argc, char *argv[]) { QCoreApplication a(argc, argv); MyThread thread; thread.start(); return a.exec(); } ``` ### 模态对话框事件循环 启动模态对话框的事件循环可以使用`QDialog::exec()`。示例代码如下: ```cpp #include <QApplication> #include <QDialog> int main(int argc, char *argv[]) { QApplication a(argc, argv); QDialog dialog; // 启动模态对话框的事件循环 dialog.exec(); return a.exec(); } ``` ### 手动强制运行事件循环 在耗时的任务中,可以通过调用`QCoreApplication::processEvents()`函数来手动强制运行事件循环。`QCoreApplication::processEvents()`函数会发出事件队列中的所有事件,并且立即返回到调用者,这实际上是模拟了一个事件循环。示例代码如下: ```cpp #include <QCoreApplication> #include <QDebug> void longRunningTask() { for (int i = 0; i < 1000000; ++i) { // 手动处理事件 QCoreApplication::processEvents(); // 模拟耗时操作 qDebug() << "Processing step" << i; } } ```
评论 1
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值