沙漠网站建设,用word做旅游网站,网站制作建设兴田德,自己开发微信小程序教程Qt 5 Alpha 已经发布。我们会在后面的文章中看到 Qt 5 的新变化。今天#xff0c;我们先来看一下 Qt 5 带来的一个最主要的变化#xff1a;signal/slot 机制的改变。Qt 5 之前的语法 在 Qt 5 之前#xff0c;我们需要使用下面的语句来链接 signal 和 slot#xff1a; 12con… Qt 5 Alpha 已经发布。我们会在后面的文章中看到 Qt 5 的新变化。今天我们先来看一下 Qt 5 带来的一个最主要的变化signal/slot 机制的改变。Qt 5 之前的语法 在 Qt 5 之前我们需要使用下面的语句来链接 signal 和 slot 12connect(sender, SIGNAL(valueChanged(QString, QString)), receiver, SLOT(updateValue(QString)));Qt 实际上利用SIGNAL和SLOT这两个宏把其后的函数名转换成一个字符串。随后moc 将会扫描全部文件将所有的 signal 和 slot 提取出来做成一个映射表。QObject::connect()函数则会从这个映射表里面找到该字符串从 signal 的名字就可以找到 slot 的名字因此也就知道了在 signal emit 的时候该去调用哪一个 slot 函数。Qt 5 之前的 signal/slot 语法的问题从上面的解释可以看出Qt 5 之前版本提供的这种语法其实有一些问题没有编译期检查因为函数名被处理成字符串所有的检查都是在运行时完成的。这就是为什么有时会发生编译通过了但 slot 并没有被调用。此时你就应该去检查 console 的输出看看有没有什么 warning 说明 connect 并没有成功。因为处理的是字符串所以 slot 中的类型名字必须用 signal 的完全一致而且在头文件中的和实际 connect 语句中的也必须一致。也就是说如果你使用了 typedef 或者 namespaceconnect 就可能不成功在 Qt 5 之前的版本中我们当然也可以使用 namespace但是必须保证头文件中的和 connect 语句中的文本完全一致。新语法使用函数指针在 Qt5 提供了一套新的语法。之前的语法依然可以使用但是现在我们有了更好的选择 1 2 connect(sender,Sender::valueChanged, receiver,Receiver::updateValue); 这个看起来和之前的版本很类似因此很容易迁移到新的语法。下面我们看看新语法有什么好处 编译器检查 如果把 signal 或者 slot 名字编写错误或者 slot 的参数同 signal 不一致你会在编译期就获得一个错误。这肯定会在重构、或者修改 signal 或 slot 的名字时节省很多时间。 另一个影响是Qt 可以利用static_cast返回更友好的错误信息。例如如果我们少了Q_OBJECT宏则会有 123456789#include QtCore/QtCoreclass Goo : public QObject { Goo() { connect(this, Goo::someSignal, this, QObject::deleteLater); }signals: void someSignal();};其错误信息是 1 2 3 4 5 qobject.h:Inmemberfunctionvoid QObject::qt_check_for_QOBJECT_macro(const T) const [with T Goo]: qobject.h:535:9: instantiatedfromstatic typename QtPrivate::QEnableIf::ArgumentCount) (int)(QtPrivate::FunctionPointer::ArgumentCount)), void*::Type QObject::connect(const typename QtPrivate::FunctionPointer::Object*, Func1, const typename QtPrivate::FunctionPointer::Object*, Func2, Qt::ConnectionType) [with Func1 void (Goo::*)(), Func2 void (QObject::*)(), typename QtPrivate::QEnableIf::ArgumentCount) (int)(QtPrivate::FunctionPointer::ArgumentCount)), void*::Type void*, typename QtPrivate::FunctionPointer::Object Goo, typename QtPrivate::FunctionPointer::Object QObject] main.cc:4:68: instantiatedfromhere qobject.h:353:5:error:voidvaluenotignoredasitoughttobe make:***[main.o]Error1 参数的自动类型转换 现在我们不仅可以更好地使用 typedef 或 namespace而且可以利用隐式类型转换。在下面的例子中我们的 signal 有一个QString参数而 slot 需要的是QVariant。在新语法中QString将被自动转换成QVariant 123456789101112class Test : public QObject{ Q_OBJECTpublic: Test() { connect(this, Test::someSignal, this, Test::someSlot); }signals: void someSignal(const QString );public: void someSlot(const QVariant );};连接到任意函数如果你留心上面的例子就会发现我们的 signal 被连接到了一个 public 函数但这个函数并不是 slot。Qt 的新语法通过函数指针直接调用函数而不需要 moc 的特殊处理但是 signal 依然需要。更进一步我们可以将 signal 连接到任意函数 1 2 3 4 5 6 staticvoidsomeFunction(){ qDebug()pressed; } // ... somewhere else QObject::connect(button,QPushButton::clicked,someFunction); 这样处理就可以让你很方便的同 boost 或者 tr1::bind 这样的第三方库协作。 C11 lambda 表达式 至此之前我们所有的示例都是基于 C98. 标准的。但是如果你的编译器支持 C11我相信你一看到“函数指针”这几个字就一定会想到 C11 的新特性Lambda 表达式。现在Lambda 表达式至少被 MSVC 2010GCC 4.5clang 3.1 这几个编译器支持。不过对于后面两个编译器你需要在编译时加上 -stdc0x 参数。 现在我们可以用 Lambda 表达式重写了 1 2 3 4 5 6 7 8 9 10 11 voidMyWindow::saveDocumentAs(){ QFileDialog *dlgnewQFileDialog(); dlg-open(); QObject::connect(dlg,QDialog::finished,[](intresult){ if(result){ QFilefile(dlg-selectedFiles().first()); // ... save document here ... } dlg-deleteLater(); }); } 这种语法允许我们更方便地编写异步代码。