QT中信号和槽用在多线程中

注意:Qt的信号与槽机制可以将任何继承自QObject类的对象捆绑在一起,使不同对象之间能够相互通信。

connect用于连接qt的信号和槽,在qt编程过程中不可或缺。它其实有第五个参数,只是一般使用默认值,在满足某些特殊需求的时候可能需要手动设置。这是用在多线程中的重点。

两个线程:

1.

class MyThread:public QThread
{
  Q_OBJECT
  ...
signals:
  void MsgSignal(const QString& tep);//用于向主线程传递字符串
protected:
  void run();//run 中的内容才是子线程中执行的内容!
}

void MyThread::run()
{
  Sleep(3000);//头文件:windows.h
  QString tep("mou-mou-mou");
  emit MsgSignal(tep);
  Sleep(3000);//各个Sleep的位置可放置具体执行的工作
}

2.

class MainApp:public QWidget
{
  Q_OBJECT
  ...
public:
  MainApp();
  ...
private slots:
  void OnMsgSignal(const QString& tep2);//接受子线程传递字符串用于显示
private:
  MyThread* m_thread;
}

MainApp::MainApp()
{
  ...
  m_thread= new MyThread();
  connect(m_thread, SIGNAL(MsgSignal(const QString&)),
              this, SLOT(OnMsgSignal(const QString&)));//此处connect的第五个参数默认变成Qt::QueuedConnection
  m_thread->start();
}

void MainApp::OnMsgSignal(const QString& tep2)
{
  //使用子线程传递来的tep2
}

1.connect函数的第五个参数代表信号与槽的连接模式,线程间的信号与槽不能使用Qt::DirectConnection直接连接方式,因为它要求在发信号的线程内执行槽函数。而Qt::QueuedConnection队列方式将信号转换成事件发送到槽函数所在线程的消息队列中让槽函数所在线程来处理,可以实现线程安全的线程间的通信。这样的时效性也不差,上面的实现中,会在子线程“run()”函数中的第二个Sleep之前执行主线程的“OnMsgSignal(constQString& tep2)”。

于是在调试的时候子线程的emit MsgSignal(const QString& tep);的下一步并不会立即跳转到主线程的OnMsgSignal(const QString& tep2),我开始还以为信号中途丢了没送达呢。。而强制使用Qt::DirectConnection模式却怎么也摆脱不了错误。

2.线程间用“信号与槽”传递引用参数的话,一定要加const,因为const文字常量存在常量区中,生命周期与程序一样的长。这样可以避免slot调用的时候参数的运行期已过而使引用无效。

第五个参数的用法:

Qt::AutoConnection: 默认值,使用这个值则连接类型会在信号发送时决定。如果接收者和发送者在同一个线程,则自动使用Qt::DirectConnection类型。如果接收者和发送者不在一个线程,则自动使用Qt::QueuedConnection类型。

Qt::DirectConnection:槽函数会在信号发送的时候直接被调用,槽函数运行于信号发送者所在线程。效果看上去就像是直接在信号发送位置调用了槽函数。这个在多线程环境下比较危险,可能会造成奔溃。

Qt::QueuedConnection:槽函数在控制回到接收者所在线程的事件循环时被调用,槽函数运行于信号接收者所在线程。发送信号之后,槽函数不会立刻被调用,等到接收者的当前函数执行完,进入事件循环之后,槽函数才会被调用。多线程环境下一般用这个。

Qt::BlockingQueuedConnection:槽函数的调用时机与Qt::QueuedConnection一致,不过发送完信号后发送者所在线程会阻塞,直到槽函数运行完。接收者和发送者绝对不能在一个线程,否则程序会死锁。在多线程间需要同步的场合可能需要这个。

Qt::UniqueConnection:这个flag可以通过按位或(|)与以上四个结合在一起使用。当这个flag设置时,当某个信号和槽已经连接时,再进行重复的连接就会失败。也就是避免了重复连接。

  • 5
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
以下是使用Python实现Qt信号多线程的启动线程和停止线程的示例代码: ```python from PyQt5.QtCore import QThread, pyqtSignal from PyQt5.QtWidgets import QApplication, QWidget, QVBoxLayout, QPushButton import time class MyThread(QThread): signal = pyqtSignal(str) def __init__(self, parent=None): super().__init__(parent) self.is_running = False def run(self): self.is_running = True while self.is_running: self.signal.emit('Thread is running') time.sleep(1) def stop(self): self.is_running = False class MainWindow(QWidget): def __init__(self): super().__init__() self.initUI() def initUI(self): self.layout = QVBoxLayout() self.btn_start = QPushButton('Start Thread') self.btn_start.clicked.connect(self.start_thread) self.layout.addWidget(self.btn_start) self.btn_stop = QPushButton('Stop Thread') self.btn_stop.clicked.connect(self.stop_thread) self.layout.addWidget(self.btn_stop) self.setLayout(self.layout) self.thread = MyThread() self.thread.signal.connect(self.show_message) def start_thread(self): self.thread.start() def stop_thread(self): self.thread.stop() def show_message(self, message): print(message) app = QApplication([]) window = MainWindow() window.show() app.exec_() ``` 在这个示例,我们创建了一个`MyThread`类,继承自`QThread`,实现了线程的启动和停止方法。然后,在`MainWindow`类,我们创建了两个按钮,分别用来启动和停止线程,并将它们的`clicked`信号与相应的函数连接起来。同时,我们还创建了一个`MyThread`类的实例,并将它的`signal`信号连接到一个函数`show_message`上,用来显示线程的运行状态。最后,我们启动了Qt应用程序的事件循环,以便响应用户的操作。 在这个示例,我们使用了`QThread`、`pyqtSignal`等Qt的基本类和信号机制,实现了一个简单的多线程应用程序。这个应用程序可以启动一个线程,并周期性地向主界面发送信息,同时也可以停止正在运行的线程
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

h1007886499

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值