在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. vue单位文本控件与vue加密文本控件

    vue单位文本控件: 使用方式: npm install dami-text-input --save 使用: <text-input v-model="test" :uni ...

  2. js转盘大抽奖 自定义概率

    公司项目搞优惠活动,让做一个转盘抽奖的活动,转盘抽奖让他转起来 按照概率停止其实都麻烦,但是概率如果设置在前端就会很大的安全漏洞,所以无论为了安全性还是后期的维护问题都要把概率写到后台配置里然后读取配 ...

  3. Django的认证系统和Django admin的简单使用

    Django自带的用户认证 我们在开发一个网站的时候,无可避免的需要设计实现网站的用户系统.此时我们需要实现包括用户注册.用户登录.用户认证.注销.修改密码等功能,这还真是个麻烦的事情呢. Djang ...

  4. 关于生物项目上的blast和viroblast

    最近要做一个跟生物有关的项目,隔行如隔山呀,好多工具以前都没听过,blast分到我头上啦,查查,查查 BLAST (Basic Local Alignment Search Tool)是一套在蛋白质数 ...

  5. 在vue中调用echarts中的地图散点图~

    首先!当然是在vue中引入echarts! 命令行  npm install echarts --save 在main.js文件中里引入        import echarts from 'ech ...

  6. HATEOAS约束

    HATEOAS(Hypermedia as the engine of application state)是 REST 架构风格中最复杂的约束,也是构建成熟 REST 服务的核心.它的重要性在于打破 ...

  7. AutoLayout的坑

    本文投稿文章,作者:MangoMade(简书) AutoLayout非常强大也非常易用,可读性也很强,加上各种第三方AutoLayout库,让你布起局来犹如绷掉链子的狗!根本停不下来!以前的 1 la ...

  8. Python OS模块常用

    python 读写.创建 文件 第二个:目录操作-增删改查 第三个:判断 第四个:PATH 第四个:os.mknod 创建文件(不是目录) import os os.chdir("/&quo ...

  9. angular6 input节流

    一直以为   pipe(debounceTime(1000), distinctUntilChanged())  不起作用 原因:使用方法错误 <input type="text&qu ...

  10. popstate事件在低版本webkit中的调用

    popstate是H5的history系列中的事件,但是在低版本的webkit中会自动触发.H5中的history api是不会使页面发生跳转的,只是操作地址栏和响应的state属性而已,而且是手动操 ...