一、declval的介绍

std::declval定义在头文件中:

template<class T>
typename std::add_rvalue_reference<T>::type declval() noexcept;

看定义它应该是返回一个右值引用(在T 是(可有 cv 限定的) void ,此情况下返回类型是 T)。在c++的文档中介绍说其可以不通过构造函数就可以使用T的成员函数,它只能用于不求值语境。这个模板函数没有具体的实现,无法调用。一般用于与decltype,sizeof等关键字配合来进行类型推导、占用内存空间(某一类型的对象的引用的占位符)计算等,一般在模板编程中应用较多。
需要注意的是,std::declval是在编译期进行处理完成的,也就是说,它在编译期过程中生成一个值对象,但是其并不会被编译为可执行期的二进制实体。或者可以这样理解,在开发过程中,有时候儿并不需要真正的拿到这个实例对象,而只是来对这个对象进行一个类型描述,如果真想用其来完成某种计算,一般外面会罩上decltype ,这也是在网上往往这两个模板函数放在一起分析的原因。这种情况是不是有些熟悉?在模板元编程里是不经常遇到这种情况,所以其在模板的元编程里应用也比较多。
下面看一个占位符的示例:

// 在下面的情况中,这里使用了std::declval,在调用运算符?:的时候就不需要调用 T1 和 T2 的(默认)构造函数
#include <utility>
template<typename T1, typename T2,
typename R = typename std::decay_t< decltype(true ? std::declval<T1>() : std::declval<T2>())> >
R GetMax (T1 a, T2 b)
{return b < a ? a : b;
}

弄明白了这个,就明白了std::declval的用处了。

二、decltype介绍

decltype可以在编译期内推导表达式所得值的类型,在上面的declval中可以看到一起使用的效果。其实看代码也可以明白,就是推导出类型结果。拿到这个类型结果,就可以搞事情了。可是这里面有一个问题,为什么它是一个关键字?这个类型推导多么简单的一个事儿。其实不然,如果后面有一大堆的表达式,复杂的不得了,你就会发现,用这个关键字可就真的好。那什么情况下会有一大串的表达式呢?仍然是元编程中多。谁也不想把一大串的模板声明不断的抄来抄去,这老麻烦了。也就是说,decltype,auto,using在某些情况下起到的作用有些类似。在前面也可以看到这些用法的使用,这里就不再做赘述。它的语法定义如下:

decltype ( 实体 )  (1) (C++11 起)
decltype ( 表达式 )    (2) (C++11 起)

decltype为什么配合std::declval一起使用,有一个重要原因就是前者需要推导类型时对象要求需要有默认构造函数,而后者不需要。这就在模板元编程中起到了重要的作用。另外,纯虚类也是一个问题,declval 可以绕开纯虚基类不能实例化的问题。decltype和逗号表达式一起工作时,需要考虑逗号表达式是从左到右依次计算,最后一个做为返回值,这个在以前的变参模板中也用到过。看一下例子:

template<typename T>
auto len (T const&& t) -> decltype((void)(t.size()) , T::size_type)
{return t.size();
}

这个size函数可以说做一个Assert,如果T中有这个函数,则替换成功,否则直接报错。这也算是一个小技巧。

三、例程

这两个功能可以在一起使用,然后创造一些小惊喜。
看一看std::declval的例程:

#include <utility>
#include <iostream>struct Default { int foo() const { return 1; } };struct NonDefault
{NonDefault() = delete;int foo() const { return 1; }
};int main()
{decltype(Default().foo()) n1 = 1;                   // n1 的类型是 int
//  decltype(NonDefault().foo()) n2 = n1;               // 错误:无默认构造函数decltype(std::declval<NonDefault>().foo()) n2 = n1; // n2 的类型是 intstd::cout << "n1 = " << n1 << '\n'<< "n2 = " << n2 << '\n';
}

是不是非常简单明了,其实学习这些东西,就得从最基础最简单的地方,把基本的知识掌握了,才能使用上面的各种变化和技巧。

再看一下decltype的例程:

#include <iostream>
#include <type_traits>struct A { double x; };
const A* a;decltype(a->x) y;       // y 的类型是 double(其声明类型)
decltype((a->x)) z = y; // z 的类型是 const double&(左值表达式)template<typename T, typename U>
auto add(T t, U u) -> decltype(t + u) // 返回类型依赖于模板形参
{                                     // C++14 开始可以推导返回类型return t+u;
}int main()
{int i = 33;decltype(i) j = i * 2;std::cout << "i = " << i << ", "<< "j = " << j << '\n';std::cout << "i 和 j 的类型相同吗?"<< (std::is_same_v<decltype(i), decltype(j)> ? "相同" : "不同") << '\n';auto f = [](int a, int b) -> int{return a * b;};decltype(f) g = f; // lambda 的类型是独有且无名的i = f(2, 2);j = g(3, 3);std::cout << "i = " << i << ", "<< "j = " << j << '\n';
}

在c++17以后,还可以std::delctype(auto)这种形式来推导相关的类型,例程可以看一下前面的拖尾类型的文章,上面有c++14和c++17两种的不同方式,就用到这种形式。
看了上面的两个应用,再看一个网上的纯虚类的应用:

#include <iostream>namespace {struct base_t { virtual ~base_t(){} };template<class T>struct Base : public base_t {virtual T t() = 0;};template<class T>struct A : public Base<T> {~A(){}virtual T t() override { std::cout << "A" << '\n'; return T{}; }};
}int main() {decltype(std::declval<A<int>>().t()) a{}; // = int a;decltype(std::declval<Base<int>>().t()) b{}; // = int b;std::cout << a << ',' << b << '\n';
}

四、总结

其实很多东西,基础的太简单,仿佛看一眼就会了。但是过了好久,又突然发现,这种简单的东西自己从来没用过。也不问为什么,反正是觉得用不到。忽然有一天,看到人家大牛的代码里这种简单的知识组合起来满飞,大脑于是一片迷茫,根本不知道怎么回事儿。回过头来看吧,又觉得基础的东西太多,不看吧,确实又不明白代码。问人吧,又没人可问。于是,大多数人可能就不了了之了。
学习在这里突然断了片儿,上,够不着;下,不甘心。怎么破除这种情况呢?还是得反过来,把基础重新牢牢打好,把大牛的复杂代码分解成一块一块的不断分析,从细节入手,除了一些特别孤僻的技巧,一般来说,都会慢慢搞定。从生到熟,从熟到初步应用,到自主应用,甚至自由组合放飞技术。这都是大有可能的。努力吧!

c++11中的declval和decltype相关推荐

  1. c++11:std::declval、decltype

    1.decltype是类型推导 #include <iostream>struct A { double x; }; const A* a;decltype(a->x) y; // ...

  2. C++11中头文件type_traits介绍

    C++11中的头文件type_traits定义了一系列模板类,在编译期获得某一参数.某一变量.某一个类等等类型信息,主要做静态检查. 此头文件包含三部分: (1).Helper类:帮助创建编译时常量的 ...

  3. C++/C++11中头文件functional的使用

    <functional>是C++标准库中的一个头文件,定义了C++标准中多个用于表示函数对象(function object)的类模板,包括算法操作.比较操作.逻辑操作:以及用于绑定函数对 ...

  4. C++/C++11中std::string用法汇总

    C++/C++11中std::string是个模板类,它是一个标准库.使用string类型必须首先包含<string>头文件.作为标准库的一部分,string定义在命名空间std中. st ...

  5. C++11中unique_ptr的使用

    在C++中,动态内存的管理是通过一对运算符来完成的:new,在动态内存中为对象分配空间并返回一个指向该对象的指针,可以选择对对象进行初始化:delete,接受一个动态对象的指针,销毁该对象,并释放与之 ...

  6. C++11中auto的使用

    在C语言中,就有了auto关键字,它被当作是一个变量的存储类型修饰符,表示自动变量(局部变量).它不能被单独使用,否则编译器会给出警告.在C++11标准中,添加了新的类型推导特性.在C ++11中,使 ...

  7. C++11中值得关注的几大变化 .

    Lambda 表达式 Lambda 表达式的形式是这样的: view plaincopy to clipboardprint? [capture](parameters)->return-typ ...

  8. C++11中值得关注的几大变化

    赖勇浩(http://laiyonghao.com) 声明:本文源自 Danny Kalev 在 2011 年 6 月 21 日发表的<The Biggest Changes in C++11( ...

  9. C++11 中值得关注的几大变化

    2019独角兽企业重金招聘Python工程师标准>>> 源文章来自前C++标准委员会的 Danny Kalev 的 The Biggest Changes in C++11 (and ...

最新文章

  1. 玩转oracle 11g(52):Oracle导出导入表(.sql、.dmp文件)两种方法
  2. 小白上手Mysql数据库指南~~
  3. Onvif开发之Linux下gsoap的使用及移植
  4. IOS开发基础知识--碎片5
  5. GWT HTML editor
  6. 谷歌和GitHub 联手提出新方法,提振软件供应链安全
  7. 发布Android程式步骤
  8. js 正则表达式匹配定义及使用
  9. nohub java -jar xx.jar /dev/null 21 以及/dev/null是什么 21又是什么?
  10. 关于win10声卡驱动正常 插入耳机小喇叭显示红叉号 且检测提示未插入耳机的问题
  11. 准备给ubuntu18.04安装杀毒软件
  12. The page at ‘xxx‘ was loaded over HTTPS, but requested an insecure resource ‘xxx‘.
  13. 【社区周会】2021-04-27 内容概要
  14. 如何删除github上的文件夹(抖机灵方法)
  15. 空中旋球计算机控制系统,自动乒乓球发球机设计及其控制系统的研究
  16. USB Type C告白
  17. java f1_JAVA编码(19)——java使用f1j9swing来生成excel文件
  18. 微信内置浏览器屏蔽网页链接怎么办,微信跳转外部浏览器的实现教程
  19. 2021 || You Cannot Easily Catch Me: A Low-Detectable Adversarial Patch for Object Detectors
  20. 腾讯天美后端2018实习一面面经

热门文章

  1. android qq很多压缩包,微信QQ总是占用手机大量内存?这次腾讯推出官方版清理工具了...
  2. 查看win10系统的CUDA版本
  3. 【修复】Word“文件发生错误”,无法保存文件
  4. 亚特兰提斯之帝国的遗址
  5. VBA,用VBA进行分列(拆分列)的2种方法
  6. 计算机电缆芯数,DJYVP22电缆|电线(直径、重量、芯数)
  7. unity将遮挡人物的模型透明化_笔记
  8. 图像翻译/Transformer:ITTR: Unpaired Image-to-Image Translation with Transformers用Transfor进行非配对图像对图像的转换
  9. 两台linux服务器互相自动备份
  10. Python实现占用栅格地图的生成(Occupancy Grid Generation)