当前位置:首页 > 芯闻号 > 充电吧
[导读]新老语法如果我们将一个QSlider对象的valueChanged信号链接到一个QSpinBox对象的setValue槽,使用传统方式:connect(slider, SIGNAL(valueChan


新老语法


如果我们将一个QSlider对象的valueChanged信号链接到一个QSpinBox对象的setValue槽,使用传统方式:

connect(slider, SIGNAL(valueChanged(int)), spinbox, SLOT(setValue(int)));

而使用新式语法,这个样子:

connect(slider, &QSlider::valueChanged, spinbox, &QSpinBox::setValue);

使用新式语法:

编译期:检查信号与槽是否存在,参数类型检查,Q_OBJECT是否存在信号可以和普通的函数、类的普通成员函数、lambda函数连接(而不再局限于信号函数和槽函数)参数可以是 typedef 的或使用不同的namespace specifier可以允许一些自动的类型转换(即信号和槽参数类型不必完全匹配)

这一切都太吸引人了,我们稍候逐一查看

例子

例子很简单,考虑到大家应该没安装该版本的Qt,故稍微罗嗦一下。

(注意:现在是2011年6月15日,或许等你感兴趣想安装时,仓库已经有了较大变化。- dbzhang800)

准备工作

仓库地址:https://qt.gitorious.org/+qt-developers/qt/qtbase-staging

你可以直接通过git克隆该仓库

git clone git://gitorious.org/+qt-developers/qt/qtbase-staging.git

然后checkout出 qobject_connect_ptr 这个分支

你可以直接下载打包后的源码 qobject_connect_ptr压缩包

剩下的工作就不用说了,configure、make、make install

代码

看个完整的程序代码,除了两个connect是新的,其他的应该都无须解释。

#include#include#include#includeclass Widget:public QWidget
{
public:
    Widget(QWidget *parent=0);
};

Widget::Widget(QWidget *parent):
    QWidget(parent)
{
    QHBoxLayout * layout = new QHBoxLayout(this);
    QSlider * slider = new QSlider(Qt::Horizontal);
    QSpinBox * spinbox = new QSpinBox;
    layout->addWidget(spinbox);
    layout->addWidget(slider);

    connect(slider, &QSlider::valueChanged, spinbox, &QSpinBox::setValue);
    connect(spinbox, static_cast(&QSpinBox::valueChanged), slider, &QSlider::setValue);
}

int main(int argc, char *argv[])
{
    QApplication a(argc, argv);
    Widget w;
    w.show();
    return a.exec();
}

恩,如你所想,程序一切正常。

老语法的不足

我们知道老式语法connect中接收的是两个字符串,

bool QObject::connect ( const QObject * sender, const char * signal, const QObject * receiver, const char * method, Qt::ConnectionType type = Qt::AutoConnection ) [static]

比如:

connect(slider, SIGNAL(valueChanged(int)), spinbox, SLOT(setValue(int)));

编译预处理以后就是:

connect(slider, "2valueChanged(int)", spinbox, "1setValue(int)");

这有什么问题呢?

即使信号和槽不存在,编译不会出问题。只有运行时会给出警告并返回false,可是大部分用户并不检查返回值。参数必须匹配,比如信号参数是 int,槽参数是 double,语法将会 connect 失败参数类型必须字面上一样,比如说都是int,但是其中一个typedef了一下:

typedef int myInt;
connect(a, SIGNAL(sig(int)), b, SLOT(slt(myInt)));

或者namespace修饰不一样

using namespace std;
connect(a, SIGNAL(sig(std::string)), b, SLOT(slt(string)));

都会导致连接失败。

我们在Qt信号和槽,与const char* 的故事一文中详细地讨论过这些问题。

新式的信号槽写法完全避免了这些问题。

编译期检查

新式语法是使用模板来实现的。由于模板的实例化是编译期完成的,所以如果有问题编译时直接就可以暴露出来,这比老式用法(问题要在运行时才能反应出来)是的巨大的改进。

信号或槽不存在

注意看connect的写法

connect(a, &Widget::sig1, b, &Widget::slt2);

都是用的函数的地址

如果相应的函数不存在,编译器将直接告知:

../newconnect/main.cpp:26:20: error: ‘sig1’ is not a member of ‘Widget’
../newconnect/main.cpp:26:41: error: ‘slt2’ is not a member of ‘Widget’

如果使用Widget的信号,而Widget中没有添加Q_OBJECT宏,编译器将直接告知

src/gui/kernel/qwidget.h: In member function ‘void QWidget::qt_check_for_QOBJECT_macro(const T&) const [with T = Widget]’:
......................
../../qt-labs/qtbase-newsignal-build/include/QtGui/../../../qtbase/src/gui/kernel/qwidget.h:149:5: error: void value not ignored as it ought to be

模板一出错,给的东西总是这麽多,只好中间大部分都省略了。

参数不匹配 如果信号参数是int,槽参数是double。或者信号参数是QString,槽参数是QVariant。将不再有问题。如果参数不能隐式cast,将会直接报错。比如信号参数是 int,槽参数是QString:

src/corelib/kernel/qobject.h: In static member function ‘static void QtPrivate::FunctionPointer::call(Ret (Obj::*)(Arg1), Obj*, void**) [with Args = QtPrivate::List, Obj = Widget, Ret = void, Arg1 = QString, Ret (Obj::*)(Arg1) = void (Widget::*)(QString)]’...

同样很的错误风格。没办法,模板总是这样子

重载的函数怎么办?

注意看我们一开始给出的例子中给出的两个 connect 语句

connect(slider, &QSlider::valueChanged, spinbox, &QSpinBox::setValue);
connect(spinbox, static_cast(&QSpinBox::valueChanged), slider, &QSlider::setValue);

前一个很简洁,可是后一个?什么情况!!!

呵呵,没办法啊,QSpinbox的valueChanged信号是重载的:

void valueChanged ( int i )
void valueChanged ( const QString & text )

只好显示调用static_cast了。

lambda函数

C++0x标准引入了lambda函数,这个东西配合新式的connect使用似乎是很有意思。

比如:当QSlider的值改变时,通过qDebug输出该值,我们只需要

connect(slider, &QSlider::valueChanged, [](int v){qDebug()<<"slider value: "<<v;});

如果在以前,我们只能先定义一个槽函数,然后connect到该槽函数。

C++0x启用 如果你使用的 MSVC 2010, 直接用,不需要任何设置如果你使用的GCC,在pro文件内添加:

QMAKE_CXXFLAGS += -std=c++0x

异步操作

lambda配合新式connect,使得异步操作变得更简单了。

打开一个创建在heap中的对话框,调用open() 不阻塞程序运行。连接其finished信号到一个lambda函数...

static void outputSelectedFileName()
{
    QFileDialog *dlg = new QFileDialog();
    dlg->open();
    QObject::connect(dlg, &QDialog::finished, [dlg, this](int result) {
        if (result) {
            QString name = dlg->selectedFiles().first();
            qDebug()<deleteLater();
    });
}



本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

VGA接口主要用于将计算机的数字图像信号转换成模拟信号,从而可以在显示器上显示。这种接口通常包含15个针脚,分成3排,每排5个孔,可以传输红、绿、蓝三种基本颜色的信号以及水平和垂直同步信号。

关键字: vga接口 信号 电压

上海2023年9月4日 /美通社/ -- 2023年8月8日,成都大运会正式落幕。来自113个国家和地区的6500名大学生运动员,在12个比赛日里同台竞技,展现青春与体育的力量。与此同时,在大运会的各个赛场、运动员村以及...

关键字: 信号 TV PS CE

瓦努阿图维拉港2023年8月28日 /美通社/ -- Vantage(或"Vantage Markets")欣然宣布在Vantage应用程序上推...

关键字: GE AN 信号 应用程序

(全球TMT2023年7月19日讯)三星电子宣布已完成其业内首款GDDR7的研发工作,年内将首先搭载于主要客户的下一代系统上验证。继2022年三星开发出速度为每秒24千兆比特(Gbps)的GDDR6 16Gb之后,GD...

关键字: DDR 三星电子 信号 GBPS

频谱分析仪是用于分析信号频谱结构的设备,可以用于测量信号的频率、幅度、功率、谱线宽度等参数,是现代电子测量和通信领域中不可或缺的工具。

关键字: 频谱分析仪 信号

频谱分析仪是用于分析信号的频率成分和功率的电子设备。分辨率是频谱分析仪的一个重要指标,它决定了频谱分析仪能够分辨的信号细节和频谱特征。在频谱分析仪的使用过程中,分辨率受到多种因素的影响,这些因素包括信号频率、信号强度、分...

关键字: 频谱分析仪 信号 分辨率

频谱分析仪是用于测量信号频率、幅度和功率谱等特性的重要工具,广泛应用于电子工程、通信、生物医学等领域。本文将介绍频谱分析仪的基本原理、使用方法和技巧,帮助读者更好地理解和使用频谱分析仪。

关键字: 频谱分析仪 信号

频谱分析仪是一种用于分析信号频率成分的重要仪器,能够在复杂信号中识别出各个频率成分,以及它们在不同时间段的强度和带宽。在科学研究、工业生产、通信网络、电子对抗等领域,频谱分析仪发挥着重要作用。

关键字: 频谱分析仪 信号

脉冲变压器是一种用于将电源信号转换为所需电压和电流的特殊变压器。它通常被用于直流电源或高频交流电源中,可用于稳压、反接保护、防过载等应用。当我们需要选型脉冲变压器时,需考虑以下几个方面:输出电压、最大输出电流、工作频率、...

关键字: 脉冲变压器 信号 电源

集成运算放大器是一种常见的电子元器件,它广泛应用于模拟电路、信号处理、控制系统等领域。集成运算放大器主要的功能是放大电压信号。它可以将微弱的输入信号放大成为符合实际需要的信号大小,同时也可以对信号进行滤波、积分、微分等操...

关键字: 运算放大器 电路 信号
关闭
关闭