c++11中的declval和decltype
创始人
2024-04-19 05:20:45
0

一、declval的介绍

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

template
typename std::add_rvalue_reference::type declval() noexcept;

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

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

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

二、decltype介绍

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

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

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

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

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

三、例程

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

#include 
#include 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().foo()) n2 = n1; // n2 的类型是 intstd::cout << "n1 = " << n1 << '\n'<< "n2 = " << n2 << '\n';
}

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

再看一下std::decltype的例程:

#include 
#include struct A { double x; };
const A* a;decltype(a->x) y;       // y 的类型是 double(其声明类型)
decltype((a->x)) z = y; // z 的类型是 const double&(左值表达式)template
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 ? "相同" : "不同") << '\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 namespace {struct base_t { virtual ~base_t(){} };templatestruct Base : public base_t {virtual T t() = 0;};templatestruct A : public Base {~A(){}virtual T t() override { std::cout << "A" << '\n'; return T{}; }};
}int main() {decltype(std::declval>().t()) a{}; // = int a;decltype(std::declval>().t()) b{}; // = int b;std::cout << a << ',' << b << '\n';
}

四、总结

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

相关内容

热门资讯

【MySQL】锁 锁 文章目录锁全局锁表级锁表锁元数据锁(MDL)意向锁AUTO-INC锁...
【内网安全】 隧道搭建穿透上线... 文章目录内网穿透-Ngrok-入门-上线1、服务端配置:2、客户端连接服务端ÿ...
GCN的几种模型复现笔记 引言 本篇笔记紧接上文,主要是上一篇看写了快2w字,再去接入代码感觉有点...
数据分页展示逻辑 import java.util.Arrays;import java.util.List;impo...
Redis为什么选择单线程?R... 目录专栏导读一、Redis版本迭代二、Redis4.0之前为什么一直采用单线程?三、R...
【已解决】ERROR: Cou... 正确指令: pip install pyyaml
关于测试,我发现了哪些新大陆 关于测试 平常也只是听说过一些关于测试的术语,但并没有使用过测试工具。偶然看到编程老师...
Lock 接口解读 前置知识点Synchronized synchronized 是 Java 中的关键字,...
Win7 专业版安装中文包、汉... 参考资料:http://www.metsky.com/archives/350.htm...
3 ROS1通讯编程提高(1) 3 ROS1通讯编程提高3.1 使用VS Code编译ROS13.1.1 VS Code的安装和配置...
大模型未来趋势 大模型是人工智能领域的重要发展趋势之一,未来有着广阔的应用前景和发展空间。以下是大模型未来的趋势和展...
python实战应用讲解-【n... 目录 如何在Python中计算残余的平方和 方法1:使用其Base公式 方法2:使用statsmod...
学习u-boot 需要了解的m... 一、常用函数 1. origin 函数 origin 函数的返回值就是变量来源。使用格式如下...
常用python爬虫库介绍与简... 通用 urllib -网络库(stdlib)。 requests -网络库。 grab – 网络库&...
药品批准文号查询|药融云-中国... 药品批文是国家食品药品监督管理局(NMPA)对药品的审评和批准的证明文件...
【2023-03-22】SRS... 【2023-03-22】SRS推流搭配FFmpeg实现目标检测 说明: 外侧测试使用SRS播放器测...
有限元三角形单元的等效节点力 文章目录前言一、重新复习一下有限元三角形单元的理论1、三角形单元的形函数(Nÿ...
初级算法-哈希表 主要记录算法和数据结构学习笔记,新的一年更上一层楼! 初级算法-哈希表...
进程间通信【Linux】 1. 进程间通信 1.1 什么是进程间通信 在 Linux 系统中,进程间通信...
【Docker】P3 Dock... Docker数据卷、宿主机与挂载数据卷的概念及作用挂载宿主机配置数据卷挂载操作示例一个容器挂载多个目...