QT高级编程技巧(一)-- 编写高效的signal & slot通信代码

关于QT的线程通信,我们都会想到signal & slot机制。先回顾下利用signal & slot机制实现控件消息处理的方法。

控件消息处理

假设我们的主界面上有一个使用ui->btn指向的QPushButton对象,要实现该对象的clicked消息处理,可以在主界面对象MainWindow上添加一个slot方法onBtnClicked,并在其构造函数中使用connect方法与ui->btn的clicked消息进行绑定,如下:

MainWindow::MainWindow(QWidget* parent) :
  QMainWindow(parent), ui(new ui::MainWindow)
{
    ui->setupUi(this);

    // 方法一
    connect(ui->btn, &QPushButton::clicked, this, &MainWindow::onBtnClicked);

    // 方法二
    //connect(ui->btn, SIGNAL(clicked), this, SLOT(onBtnClicked));

    …
}

void MainWindow::onBtnCliced(void){  doSomething();}

事实上这种方法很繁杂。单纯为了处理一个按钮的消息我们就要新建一个slot方法,当界面控件多了的时候,会使得代码臃肿不堪(即使你可以设计成多个控件消息共享一个槽函数)。

在新版的支持C++11的QT中,我们可以使用Lambda函数优雅地解决这个问题。如下:

MainWindow::MainWindow(QWidget* parent) :
    QMainWindow(parent), ui(new ui::MainWindow)
{
    setupUi(this);

    // 使用lambda函数实现slot方法
    connect(ui->btn, &QPushButton::clicked, [&](){
        doSomething();
    });
    …
}

使用Lambda函数的另一个好处是,可以借助闭包的概念加速开发。比如我曾做过一个运动控制项目,界面上有32个之多的QCheckBox控件,如下:

需要响应每个QCheckBox控件的clicked消息,改变对应的IO口电平输出。我的做法是,对这些控件进行Tab排序,并将第一个控件命名为ui->ioChk,然后编码如下:

typedef QCheckBox   *F_QCB;
#define F_QNEXT(w)  (w->nextInFocusChain())

…

MainWindow::MainWindow(QWidget* parent) :  QMainWindow(parent), ui(new ui::MainWindow){  setupUi(ui);    …  F_QCB cb = ui->ioChk;  for (int i = 0; i < 32; ++i)  {    connect(cb, &QCheckBox::clicked, [=](bool s){      setIo(i, s);  // 改变第i个端口的输出电平为s    });    cb = F_QNEXT(cb);  }  …}

要读懂上面的程序,你需要一些关于C++11标准的Lambda函数的知识,代码中使用值捕获方式捕获i变量值形成闭包函数,这些闭包函数被绑定到不同的QCheckBox的clicked消息中。

线程间通信

很长的时间里,我都以为QT的signal & slot机制只适用于单向异步通信。事实上,它可以设计为带返回的同步通信。

有个案例如下:工作线程在运行时需要同步提取界面上的参数信息(假设要提取界面上的QSpinBox的值),要同时保证界面的响应和工作线程的不间断运转。

一种可能的实现方法是,在工作线程上的某个对象定义一个request消息,绑定到MainWindow的response槽中,然后在MainWindow中定义一个answer消息,绑定到工作线程上的某个对象的onAnswer槽中。通信过程如下:

1. 工作线程需要界面参数值,发送request消息

2. 主线程接收到request消息,开始执行response槽方法

3. 在response函数中,获取界面控件的值,发送answer消息

4. 工作线程收到answer消息,调用onAnswer槽方法,恢复之前的运行流程

这种方法条理清晰,但编码实现过于繁琐。我的实现方法是,在工作线程实现一个形如request(QString req, QVariant& arg)的消息,然后以QT::BlockBlockingQueuedConnection方式连接至MainWindow的response槽方法中。编码如下:

WorkerThread声明头文件:

#ifndef __WORKER_THREAD_H#define __WORKER_THREAD_H

class WorkerThread : public QThread {public:  …  // @Override  void run();

private:  // 用于通信  class InThreadObject;  InThreadObject *ito;  …}

#endif

WorkerThread实现:

class WorkerThread::InThreadObject : public QObject {  Q_OBJECT

signals:  void request(QString req, QVariant& arg);}

void WorkerThread::run(){  ito = new InThreadObject;  ……  // 注意下面的连接使用了QT::BlockingQueuedConnection选项  connect(ito, &InThreadObject::request, MainWindow::response,  ……  // 同步获取界面参数值  QVariant var;  ito->request("param.level", var);  // 这里会同步等待主线程执行response函数完  int param = var.toInt();  ……}

MainWindow相关实现:

void MainWindow::response(QString req, QVariant& ans){  if (req == "param.level")  {    // 获取控件值并存放至ans引用变量中    ans.setValue(ui->levelSpin->value());  }}

本次经验分享完毕,关于一些具体的技术细节,如QT的消息与槽机制、C++11的Lambda函数还请读者另行学习。

本文链接:http://www.cnblogs.com/wenris/p/4447481.html

作者:wenris;联系:<[email protected]>

后会有期哦 O(∩_∩)O

时间: 2024-10-26 11:54:09

QT高级编程技巧(一)-- 编写高效的signal & slot通信代码的相关文章

QT高级编程技巧(二)-- 编写多线程和并发应用

学习QT多线程编程之前,有必要先熟悉事件循环的概念.先看一个单线程界面程序的主函数代码: int main(int argc, char* argv[]) { QApplication app(argc, argv); // 构造主窗口对象并显示 MainWindow w; w.show(); // 进入事件循环 return app.exec(); } 在程序初始化完成后,主线程进入main()函数开始执行应用代码.一般地,我们在主线程上构建界面对象,然后进入事件循环以处理控件绘制.用户输入.

js学习笔记-编写高效、规范的js代码-Tom

编写高效.规范的js代码: 1.变量命名空间问题,尽量使用局部变量,防止命名冲突(污染作用域中的全局变量):全局空间命名的变量可以在对应的文档域任意位置中使用window调用. 2.尽量使用单var定义变量(作用域开始先申明并赋值变量,便于后边使用),使用var定义的变量只作用于对应的作用域中,如定义的全局变量作用于全局作用域,函数中定义的变量作用于该局部作用域中.未用var定义的变量相当于一个全局变量,在函数中出现的该类变量作用域全局域.(但是var定义的全局变量不能用delete删除,而未定

Python高级编程技巧实战 基于Python项目与面试题讲解

精选50个Python各领域开发及面试常见问题作为训练任务,每个任务先提出问题,然后分析问题.并给出高效的解决办法,最后手把手带你解决问题,全面提升用Python快速解决问题与高效编程的能力. ----------------------课程目录------------------------------讲师:程序员硕 Linux系统工程师Freescale半导体公司Linux Kernel工程师, 对开源Linux Kernel的mtd模块贡献过多个patch. 在清华大学信研院工作期间, 参

python高级编程技巧

http://blog.sina.com.cn/s/blog_a89e19440101fb28.html Python列表解析语法[]和生成 器()语法类似 [expr for iter_var in iterable] 或 [expr for iter_var in iterable if cond_expr] 例子:[i for i in range(10)if i %2==0] Enumerate >>> i=0 >>> s=['a','b','c'] >&

Python高级编程技巧(转)

译文:http://blog.jobbole.com/61171/ 本文展示一些高级的Python设计结构和它们的使用方法.在日常工作中,你可以根据需要选择合适的数据结构,例如对快速查找性的要求.对数据一致 性的要求或是对索引的要求等,同时也可以将各种数据结构合适地结合在一起,从而生成具有逻辑性并易于理解的数据模型.Python的数据结构从句法 上来看 非常直观,并且提供了大量的可选操作.这篇指南尝试将大部分常用的数据结构知识放到一起,并且提供对其最佳用法的探讨. 推导式(Comprehensi

python 的一些高级编程技巧

正文: 本文展示一些高级的Python设计结构和它们的使用方法.在日常工作中,你可以根据需要选择合适的数据结构,例如对快速查找性的要求.对数据一致性的要求或是对索引的要求等,同时也可以将各种数据结构合适地结合在一起,从而生成具有逻辑性并易于理解的数据模型.Python的数据结构从句法上来看非常直观,并且提供了大量的可选操作.这篇指南尝试将大部分常用的数据结构知识放到一起,并且提供对其最佳用法的探讨. 推导式(Comprehensions) 如果你已经使用了很长时间的Python,那么你至少应该听

C++Builder 高级编程技巧

1. C++ Builder 网上资源 C++ Builder 研究 http://www.ccrun.com/ C++ Builder 程序员 http://mybcb.diy.myrice.com/ 电脑学习 C++ Builder 版块 http://www.itcomputer.com.cn/Programs/C/CB/ c++ Builder 快捷键大全: http://www.itcomputer.com.cn/Programs/C/CB/200512/11211.html C++

Linux Shell 高级编程技巧4----几个常用的shell脚本例子

4.几个常用的shell脚本例子    4.0.在写脚本(同样适用在编程的时候),最好写好完善的注释    4.1.kill_processes.sh(一个杀死进程的脚本) #!/bin/bash current_PID=$$ ps -aux | grep "/usr/sbin/httpd" | grep -v "grep" | awk '{print $2}' > /tmp/${current_PID}.txt for pid in `cat /tmp/${

Linux Shell 高级编程技巧3----运行级别脚本介绍

3.运行级别脚本介绍    3.1.运行级别        运行级别介绍:            0    关机            1    单用户模式            2    多用户模式,没有NFS服务            3    多用户模式            4    目前还没有使用            5    X windows 的启动模式            6    重启计算机        运行级别对应的目录(/etc/rcN.d,N是数字0.1.2...)