Qt 多线程之间的通信 实例 emit

问题描述:界面线程MainApp为主线程,工作线程MyThread为一子线程,从工作线程向主线程传递位置用于改变主线程中按钮的PushButton位置。

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

一下是几个实现的文件

// mythread.h

#ifndef MYTHREAD_H
#define MYTHREAD_H
 
#include <QThread>
#include <QPushButton>
 
class MyThread : public QThread
{
    Q_OBJECT
public:
    explicit MyThread(QPushButton *pb);
    ~MyThread();
    QPushButton *pb;
signals:
    void change_postion(int); // 用于向主线程发送数据的信号函数, 在.cpp中,没有进行定义,直接使用emit进行数据的发送
 
 
protected:
    void run();
 
public slots:
 
private:
 
 
};
 
#endif // MYTHREAD_H
 

// mythread.cpp

#include "mythread.h"
#include <QDebug>
 
MyThread::MyThread(QPushButton *pb) :
    QThread()
{
    this->pb = pb;
}
 
void MyThread::run()
{
    int i = 0;
    while(true)
    {
        qDebug("The number is : %d", i);
        i += 10;
        emit change_postion(i); // 多线程间的通信 将i的值发送到主线程中
        this->msleep(1000);
        if(i == 400)
        {
            i = 0;
        }
    }
}
 
MyThread::~MyThread()
{
}
 

// MainWindow.h

#ifndef MAINWINDOW_H
#define MAINWINDOW_H
 
#include <QtGui/QMainWindow>
#include "mythread.h"
 
class MainWindow : public QMainWindow
{
    Q_OBJECT
 
public:
    MainWindow(QWidget *parent = 0);
    ~MainWindow();
    MyThread *thread;
    QPushButton *pb;
 
private slots:
    void move_button(int i);
 
};
 
#endif // MAINWINDOW_H
 

// MainWindow.cpp

#include "mainwindow.h"
#include <QDebug>
 
MainWindow::MainWindow(QWidget *parent)
    : QMainWindow(parent)
{    
    this->setGeometry(0, 0, 400, 400);
    pb = new QPushButton("AA", this);
    thread = new MyThread(pb);
    // 将子线程中的信号 与 主线程的反应槽函数进行连接
 
    connect(thread, SIGNAL(change_postion(int)), this, SLOT(move_button(int)), Qt::QueuedConnection); // connect函数的重载,最后一个参数需要注意,不能使用默认
 
    thread->start();
}
 
MainWindow::~MainWindow()
{
    delete thread;
}
 
void MainWindow::move_button(int i)
{
    pb->move(i, i);
}
 

// main.cpp

#include <QtGui/QApplication>
#include "mainwindow.h"
 
int main(int argc, char *argv[])
{
    QApplication a(argc, argv);
    MainWindow w;
    w.show();
 
    return a.exec();
}
 

实现过程中遇到过的问题

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


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

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


我开始没注意到需要加const,程序运行时并没有报告错误,但是主线程毫无使用子线程传递来的字符串的迹象,这让我更加错误地以为子线程发送的信号丢了。


  • 13
    点赞
  • 56
    收藏
    觉得还不错? 一键收藏
  • 4
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值