在qglobal.h中,定义了很多宏。下面这个QT_REQUIER_CONFIG,展开成:

 #define QT_REQUIRE_CONFIG(feature) Q_STATIC_ASSERT_X(QT_FEATURE_##feature == 1, "Required feature " #feature " for file " __FILE__ " not available.")

  注意宏定义连字符##的作用,看来Q_STATIC_ASSERT_X是一个判断句,如果feature被配置了的话,就没有问题,相反如果没有的话,就报一个错误。

  就在同一个文件中,可以看到这个宏的定义:

 #define Q_STATIC_ASSERT_X(Condition, Message) Q_STATIC_ASSERT(Condition)

  继续寻根,还好就在同一个文件的上面一点就找到了:

 #define Q_STATIC_ASSERT(Condition) \
enum {Q_STATIC_ASSERT_PRIVATE_JOIN(q_static_assert_result, __COUNTER__) = sizeof(QStaticAssertFailure<!!(Condition)>)}

  看到了如此复杂的宏,不觉虎躯一震。继续往上找:

 #define Q_STATIC_ASSERT_PRIVATE_JOIN(A, B) Q_STATIC_ASSERT_PRIVATE_JOIN_IMPL(A, B)

  再解一次,终于到头了:

 #define Q_STATIC_ASSERT_PRIVATE_JOIN_IMPL(A, B) A ## B

  其实就是将A和B连起来。

  而对于赋值号后面的那个部分sizeof(QStaticAssertFailure<!!(Condition)>)},可以继续找到模板类的定义:

 template <bool Test> class QStaticAssertFailure;
template <> class QStaticAssertFailure<true> {};

  如果Condition存在的话,也就是被定义过的话,那么!!(Condition)就是1,对应上面的第二行,没有出现问题;否则是0,对应上面的第一行,第一行的意思是声明一个模板类,但是没有定义,所以sizeof为0。为了验证上面两句话的作用,可以新建一个工程,对这两句分别测试:

 #include <QCoreApplication>
template<bool Test> class QStaticAssertFailure;
template <> class QStaticAssertFailure<true> {};
int main(int argc, char *argv[])
{
QCoreApplication a(argc, argv);
sizeof(QStaticAssertFailure<>);
sizeof(QStaticAssertFailure<>);
return a.exec();
} template<bool Test> class QStaticAssertFailure;
  1.   如果,注释掉第2行和第7行,不能通过,理由是QStaticAssertFailure并非类模板;
  2. 当注释掉第三行时,第七第八行都会报incomplete type错误;
  3. 当无注释时,第7行报incomplete type错误;
  4. 当注释掉第7行,编译通过。

  第3行的部分其实是一个explicit specialization,显式具体化。以下观点来自Typecool的博客:

  具体化函数定义——显式具体化(explicit specialization)

  当遇到特定类型参数,需要改变函数模板定义时,可以使用该方法。

    以下实现需要 <第三代具体化(ISO/ANSI C++标准)>支持:

      1、对于给定的函数名,可以有非模板函数、模板函数和显式具体化模板函数以及它们的重载版本。
      2、显示具体化的原型和定义应以template<>打头,并通过名称来指出类型。
      3、具体化将覆盖常规模板,而非模板函数将覆盖具体化和常规模板。(表明常规函数处于最高优先级,之后是显式具体化,最后是常规模板。)

  c++的强大易用性的背后,确实有语法冗杂的问题。上面第三行template <> class QStaticAssertFailure<true> {};这里确实是一个显式具体化,它是针对于当Test是true的时候的一种具体定义。有了这个,在第八行就不会报错了

QT_REQUIER_CONFIG的更多相关文章

随机推荐

  1. python中的mysql数据库like模糊查询

    %在python中是个特殊的符号,如%s,%d分别代表了字符串占位符和数字占位符. 大家知道,mysql的模糊查询也需要用到%. 所以,可以先把需要查的字符串抽出来,再以参数方式传入. args = ...

  2. 使用vue实现自定义搜索功能

    实现效果如:http://www.ligerui.com/demos/filter/filter.htm 代码: <%@ Page Language="C#" AutoEve ...

  3. 《SQL 基础教程》第八章:SQL 高级处理

    本章分为两个部分: 窗口函数 GROUPING 运算符 它们用于以「窗口」为单位的排序.计算总和等任务. OLAP 函数 OLAP 定义:OLAP 是 OnLIne Analytical Proces ...

  4. mysql5.7 timestamp错误:there can be only oneTIMESTAMP column with CURRENT_TIMESTAMP in DEFAULT or ON UPDATE

    #1293 - Incorrect table definition; there can be only oneTIMESTAMP column with CURRENT_TIMESTAMP in ...

  5. 阿里云 RDS for MySQL支持什么引擎

    问题:我们的服务器是买的是阿里云,mysql版本5.011 ,本地和服务器配置一样,在本地可以安装discuzX3.4,但是在服务器上却报错了,如下图: 找了半天,才知道阿里云RDS 支持的mysql ...

  6. php获取id

    private static function getClientIp() { if (getenv('HTTP_X_FORWARDED_FOR')) { $tmp = explode(',', ge ...

  7. gevent模块学习(三)

    3. Group类,常用于不限制数量的管理异步任务的分组且可搜集运行结果 g = gevent.pool.Group(*args) -> Group 说明: 创建一个组对象,其实就是一个不限gr ...

  8. java 编译

    package javacodeforstudy.testcode; public class Helloworld{ public static void main(String[] args) { ...

  9. c# 使用Renci.SshNet.dll操作SFTP总结

    1.操作类 /// <summary> /// SFTP操作类 /// </summary> public class SFTPHelper { #region 字段或属性 p ...

  10. Win10系列:C#应用控件进阶5

    多线形 多线形和多边形类似,不同点在于多线形中最后一个点和第一个点不会默认被连接.在多线形的点集中,可以存在同一个开始点和终结点因而会定义成闭合图形.下面将演示如何使用Polyline控件绘制一个多线 ...