在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. Linux基础命令mkdir

    mkdir  /home/ceshi   home目录下创建测试目录 mkdir  -p  /home/ceshi1/ceshi home目录下递归创建 mkdir     /home/ceshi2  ...

  2. select top 1 和select top 1 with ties * from SC 的区别

    select top1 : * FROM SC ORDER BY score DESC 运行结果如下: sc表数据查询select top 1 S# C# Score 001 03 120 WITH ...

  3. 解决Mac应用程序软件不出现在Launchpad里面的方法

    新装了几个软件,可是打开Lauchpad之后却在里面找不到,真是烦人!然后尝试了以下方法: 1.重启电脑,没用: 2.尝试打开“应用程序(英文名称:Applications)”并找到安装的软件,然后直 ...

  4. JVM中的垃圾回收器及垃圾收集算法描述

    首先需要了解下JVM(Java虚拟机)中的内存分配情况: 收集器的介绍: Serial收集器:是最原始的收集器,是单线程的,实现简单,但是在后台收集垃圾的时候,其他的工作线程都会停止,直到垃圾收集线程 ...

  5. css--颜色值

    首先,#000000格式的颜色被成为十六进制颜色码: 6位数分为三组,每两位数一组,依次是红.黄.蓝颜色的强度: #000000可以缩写为#000:黑色 其他类推

  6. laravel的测试工具debug安装:

    在项目根目录执行: composer require barryvdh/laravel-debugbar --dev

  7. SQL Join 与 In的效率

    今天在优化朋友的一个系统, 主要他们前期是叫人外包写的, 越来越慢, 导出订单明细时, 基本都是TimeOut, 我查看到这里面是这样写: select * from Orders where ID ...

  8. Excel文件数据导入到后台保存倒数据库

    后台代码数据解析: 方法一: (简单点) import org.apache.poi.hssf.usermodel.HSSFCell;import org.apache.poi.hssf.usermo ...

  9. 强大的金融类图表库 TradingView 使用分享

    这段时间刚好做币圈交易所,运用到了现在最火的金融类图表库 -- TradingView ,就是强大,基本上现在的火币网(https://www.huobi.com),币安网(https://www.b ...

  10. 第三方jar包导入unity

    关于第三方SDK接入Unity工程方面,有许多坑,下面我把遇到的问题进行总结,希望能够帮到有需要的朋友们.1.把第三方SDK导入Eclipse遇到的问题.eclipse配置完成右键工程后没有andro ...