用编程改变大众。。。
分类: C/C++
2015-04-11 10:54:23
Qt的信号与槽机制可以将任何继承自QObject类的对象捆绑在一起,使不同对象之间能够相互通信。
成功的实现:
工作线程:
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的位置可放置具体执行的工作
}
主线程:
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调用的时候参数的运行期已过而使引用无效。
我开始没注意到需要加const,程序运行时并没有报告错误,但是主线程毫无使用子线程传递来的字符串的迹象,这让我更加错误地以为子线程发送的信号丢了。。
想不清楚的问题总算是解决了,前几天愤怒抓狂的“我”们,你们可以安心休息啦让我沿着你们没有走完的路继续前行吧!