【C++深陷】之“decltype”[通俗易懂]

【C++深陷】之“decltype”[通俗易懂]decltype被称作类型说明符,它的作用是选择并返回操作数的数据类型。随着程序越来越复杂,有时候根本搞不清到底需要什么类型,不得不回过头去从上下文进行推测。decltype可以作用于变量,也可以作用于表达式。

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全系列IDE使用 1年只要46元 售后保障 童叟无欺

0. decltype关键字

decltype被称作类型说明符,它的作用是选择并返回操作数的数据类型。

// sum的类型就是函数f返回的类型
decltype(f()) sum = x;

回想【C++深陷】之“类型与变量”第4节,我们将定义变量的方法规范为:

类型说明符 声名符列表

decltype就是一种类型说明符,它的出现主要是解决复杂的类型声明。

随着程序越来越复杂,程序中用到的类型也越来越多,这种复杂性体现在两个方面。

一是一些类型难于“拼写”,它们的名字既难记又容易写错,还无法明确体现其真实目的和含义。

二是有时候根本搞不清到底需要的类型是什么,程序员不得不回过头去从程序的上下文寻求帮助。

解决问题一,可以使用类型别名技术。

解决问题二,可以使用auto和本文的主题:decltype。

关于decltype,需要注意:

  1. 工作原理
  2. decltype + 变量 var
  3. decltype + 表达式 expr
  4. decltype + 函数名 func_name

1. 工作原理

decltype并不会实际计算表达式的值,编译器分析表达式并得到它的类型。

函数调用也算一种表达式,因此不必担心在使用decltype时真正的执行了函数,正如前例中的f()

2. decltype + 变量

根据【C++深陷】之“表达式与运算符”中关于表达式的定义,我们知道单独使用一个变量,相当于一个最简单的表达式。

但是在decltype这里有区别。

①当使用decltype(var)的形式时,decltype会直接返回变量的类型(包括顶层const和引用),不会返回变量作为表达式的类型。

const int ci = 0, &cj = ci;

// x的类型是const int
decltype(ci) x = 0;

// y的类型是const int &
decltype(cj) y = x;

decltype加指针也会返回指针的类型。

decltype加数组,不负责把数组转换成对应的指针,所以其结果仍然是个数组(P206)。

总之decltype(var)完美保留了变量的类型。

“变量作为表达式的类型”是什么呢?见下节。

3. decltype + 表达式

②当使用decltype(expr)的形式时,decltype会返回表达式结果对应的类型。

回想【C++深陷】之“表达式与运算符”,一个表达式的结果不是左值,就是右值

关于左值和右值,请参考【C++深陷】之“左值与右值”

因此,decltype(expr)的结果根据expr的结果不同而不同:expr返回左值,得到该类型的左值引用;expr返回右值,得到该类型。

int i = 42, *p = &i, &r = i;

// r + 0是一个表达式
// 算术表达式返回右值
// b是一个int类型
decltype(r + 0) b;

// c是一个int &
decltype(*p) c = i;

这里注意一下decltype(*p)

解引用运算符*作用于指针类型,得到了p指向的对象的左值(*p = 2很正确),p是指向int的指针,因此decltype(*p)得到的类型是int &

再来看第2节说到的“变量作为表达式的类型”。

当变量作为表达式时,返回的是该变量的一个左值形式(因为该表达式的结果可以作为赋值语句的左侧的值)。因此,使用decltype理应得到一个该类型的左值引用。

但是decltype单独作用于对象,没有使用对象的表达式的属性,而是直接获得了变量的类型。

要想获得变量作为表达式的类型,可以加一个括号:

int i = 42;

// 加了括号,变成了表达式
// 返回的是i的左值形式
// 因此ri的类型是int &
decltype((i)) ri = i;

那么请问,下面的代码,temp是什么类型:

int i = 42, *p = &i;

decltype((p)) temp = p;

解析:decltype作用的是表达式,(p)得到的是p的左值,所以temp一定是一个引用;p是指向int类型的指针,因此decltype得到的是指向int类型的指针的引用

4. decltype + 函数

C++中通过函数的返回值和形参列表,定义了一种名为函数类型的东西。它的作用主要是为了定义函数指针

例如:

// 声明了一个函数类型
using FuncType = int(int &, int);

// 下面的函数就是上面的类型
int add_to(int &des, int ori);

// 声明了一个FuncType类型的指针
// 并使用函数add_to初始化
FuncType *pf = add_to;

int a = 4;

// 通过函数指针调用add_to
pf(a, 2);

我们可以使用decltype获得函数add_to的类型:

decltype(add_to) *pf = add_to;

这样的声明就简单多了。

③当使用decltype(func_name)的形式时,decltype会返回对应的函数类型,不会自动转换成相应的函数指针。

5. 总结

decltype是为了解决复杂的类型声明而使用的关键字,称作decltype类型说明符

decltype可以作用于变量、表达式及函数名。①作用于变量直接得到变量的类型;②作用于表达式,结果是左值的表达式得到类型的引用,结果是右值的表达式得到类型;③作用于函数名会得到函数类型,不会自动转换成指针。

decltype不会去真的求解表达式的值,可以放心使用。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/195408.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • mybatis 设置jdbcType与JavaType和JdbcType对应表

    mybatis 设置jdbcType与JavaType和JdbcType对应表使用mybatis对mysql数据库进行增删改查时,如果传递的参数存在空值,那么会报空值异常: select*fromuserwhere1=1and <iftest="params.userCode!=nullandparams.userCode.trim()!=”">anduserCode=#{params.userCode}…

    2022年10月20日
    2
  • 关于旁路由设置后,主路由WIFI无法上网的问题「建议收藏」

    关于旁路由设置后,主路由WIFI无法上网的问题「建议收藏」前言旁路由设置好后,手机、电脑连接主路由WIFI,会无法访问外网。但是,如果电脑用网线连接主路由,则可以正常上网。这究竟是怎么一回事儿呢?1.旁路由解释旁路由:旁路由其实并不是路由,路由是用来连接不同网络的,最常用的就是用来连接互联网和局域网。旁路由起到的主要是网关的作用,是用来分流数据和扩展插件的。因此,严谨一点的叫法应该是旁路网关,只是大家好像约定俗成了都叫做旁路由,所以我们这里也跟着叫旁路由,但是要明白它的核心是网关而不是路由。2.网络流量示意图如图所示,对于普通流量,由于旁路

    2022年6月12日
    117
  • idea如何查找替换_wps表格怎么查找替换文字

    idea如何查找替换_wps表格怎么查找替换文字在平时敲代码的时候经常碰到,咦,这个变量名好像不太合适,但又写了好多这时候可以怎么办呢?Pycharm里面给我们准备了替换功能————–windows电脑—————1.Ctrl+r替换2.Ctrl+Shift+F全局查3.Ctrl+Shift+R全局替换————–MAC电脑—————1.command+F全局查找2.command+R全局替换…

    2022年8月25日
    14
  • jQuery each方法中结束本次循环和退出循环

    jQuery each方法中结束本次循环和退出循环

    2022年3月13日
    44
  • js数组截取方式splice()和slice()方法

    js数组截取方式splice()和slice()方法js数组截取方式splice()和slice()方法1.splice()splice()方法可以添加元素、删除元素,也可以截取数组片段。删除元素时,将返回被删除的数组片段,因此可以使用splice()方法截取数组片段//传递一个参数,则该方法仅执行删除操作,参数值指定删除元素的起始下标(包含该下标元素)//splice()方法将删除后面所有元素vara=[1,2,3,4,5];//定义数组varb=a.splice(2);//从第三个元素开始执行删除console

    2022年5月25日
    44
  • Cisco AAA 认证 总结「建议收藏」

    Cisco AAA 认证 总结「建议收藏」Authentication认证一:使用本地数据库R1>enR1(config)#aaanew-model(打开aaa认证)R1(config)#aaaauthenticationlogindefaultlocal配置任何登入采用Local本地用户数据库R1(config)#usernamebd…

    2022年6月1日
    37

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号