c++学习之auto和decltype篇

2024-05-05 18:18
文章标签 c++ 学习 auto decltype

本文主要是介绍c++学习之auto和decltype篇,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

auto篇

引入: 早在c++11以后,auto这个新的关键词就会定义出来了,它本质目的是为了简化复杂的一些类的定义,让编译器自己根据变量、表达式、函数参数等等的类型来判断需要定义的变量的类型

下面拿代码说话,带你们看看究竟auto是什么东西!(平时绝对不会这么用的):

	auto i = { 100 };auto d = { 5.1231 };cout << "i类型为" << typeid(i).name() << endl;cout << "d类型为" << typeid(d).name() << endl;

来看一下结果:
在这里插入图片描述
总结: 我们关注后面,一个int,一个double,没有问题!

auto之缺陷

那究竟有什么缺陷呢?

接下来引入一段代码来说明问题:

	const int a{ 10 }; int aa{ 10 };auto b{ a };int& c{ aa };auto d{ c };cout << "b类型为" << typeid(b).name() << endl;cout << "d类型为" << typeid(d).name() << endl;

我们来看一下结果:
在这里插入图片描述
追加: 惊喜不??明明是const int 却变成int ,而int& 也变成int,这足以说明了auto的问题:

  1. 不保留const属性
  2. 不保留引用属性

由于auto的这种缺陷导致我们如果出现一下写法时,就出问题了!

	auto Max(int& a, int& b){return a > b ? a : b;}int main(){int a{ 10 }, b{ 20 };auto c = Max(a, b);cout << "c=" << c << "  c类型为" << typeid(c).name() << endl;}

来看看结果:
看起来好像除了没有返回引用类型int,结果也对的。没啥事呀?真实情况呢!你返回的既不是a也不是b,而是一个临时变量!问题的危害就很大了!!!
在这里插入图片描述
这时候需要什么来帮助我们呢?
请看下文↓

拖尾函数与decltype

拖尾函数

什么叫拖尾函数?我也不知道该怎么说它!举个代码你就懂了:

	auto Max(int& a, int& b)->int&   //是不是有个小尾巴呢?很形象把{return a > b ? a : b;}

//这个小尾巴就可以指定我们的函数返回类型了,是不是很棒呢?

decltype

什么叫decltype?只能说有点像auto的功能,但有区别并且更好用!

来看一下如何使用的呢:

	int a{ 10 };unsigned b{ 10 };const int c{ 10 };int& d{ a };int* e{ &a };decltype(a) x1; //相当于int x1decltype(b) x2; //相当于unsigned x2decltype(c) x3=c; //相当于const int x3=cdecltype(d) x4=x1; //相当于int& x4=x1decltype(e) x5=e; //相当于int* x1=e

总结: 是不是感觉很好用呢?它弥补了auto丢失const和引用的缺陷

注意: 当然在使用decltype时,也会有格外需要关注的点!当decltype(xxx)xxx发生了运算时,会出现怎样的情况呢?

案例1如下:

	int a{ 10 }, b{ 20 };int* c{ &a };int d[10]{};decltype(a + b) x0; //相当于int x0decltype(*c) x1=a; //相当于int& x1=adecltype(d[5]) x2 = a; //相当于int& x2=a

自己粗俗的总结一下: 当decltype()括号中出现了运算时,根据运算出结果是否有固定的内存地址,若有则判定为引用类型,否则判定为值类型;

案例2如下:

int avg1(int a, int b)
{cout << "auto执行了" << endl;return (a + b) / 2;
}
int avg2(int a, int b)
{cout << "decltype执行了" << endl;return (a + b) / 2;
}
int main()
{auto x1 = avg1(1, 2);   //函数会被真的执行一次decltype (avg2(1, 2)) x=10; //不会执行

在这里插入图片描述

总结一下: decltype并不会真的执行表达式的运算

decltype运用场景

场景1:

auto Max(int& a, int& b)->decltype(a > b ? a : b)
{return a > b ? a : b;
}int main()
{int a{ 10 }, b{ 20 };int c{ 30 };Max(a, b) = c;  //返回值不是引用不可能这样写的cout << "a=" << a << endl;cout << "b=" << b << endl;
}

来看一下结果:
结果肯定时不可能错的,因为函数返回值都会赋值了,肯定是左值!说明返回的是引用
在这里插入图片描述

场景2类似(c++14以后):

decltype(auto)  Max(int& a, int& b)   //是不是发现c++的牛逼之处了呢???哈哈哈
{return a > b ? a : b;
}

感谢自己努力拼搏的学习!!!老铁们觉得有点用的话,记得点点赞哦!!!我是航行的小土豆!!!

这篇关于c++学习之auto和decltype篇的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/962347

相关文章

c++ 类成员变量默认初始值的实现

《c++类成员变量默认初始值的实现》本文主要介绍了c++类成员变量默认初始值,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录C++类成员变量初始化c++类的变量的初始化在C++中,如果使用类成员变量时未给定其初始值,那么它将被

C++中NULL与nullptr的区别小结

《C++中NULL与nullptr的区别小结》本文介绍了C++编程中NULL与nullptr的区别,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编... 目录C++98空值——NULLC++11空值——nullptr区别对比示例 C++98空值——NUL

C++ Log4cpp跨平台日志库的使用小结

《C++Log4cpp跨平台日志库的使用小结》Log4cpp是c++类库,本文详细介绍了C++日志库log4cpp的使用方法,及设置日志输出格式和优先级,具有一定的参考价值,感兴趣的可以了解一下... 目录一、介绍1. log4cpp的日志方式2.设置日志输出的格式3. 设置日志的输出优先级二、Window

从入门到精通C++11 <chrono> 库特性

《从入门到精通C++11<chrono>库特性》chrono库是C++11中一个非常强大和实用的库,它为时间处理提供了丰富的功能和类型安全的接口,通过本文的介绍,我们了解了chrono库的基本概念... 目录一、引言1.1 为什么需要<chrono>库1.2<chrono>库的基本概念二、时间段(Durat

C++20管道运算符的实现示例

《C++20管道运算符的实现示例》本文简要介绍C++20管道运算符的使用与实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录标准库的管道运算符使用自己实现类似的管道运算符我们不打算介绍太多,因为它实际属于c++20最为重要的

Visual Studio 2022 编译C++20代码的图文步骤

《VisualStudio2022编译C++20代码的图文步骤》在VisualStudio中启用C++20import功能,需设置语言标准为ISOC++20,开启扫描源查找模块依赖及实验性标... 默认创建Visual Studio桌面控制台项目代码包含C++20的import方法。右键项目的属性:

c++中的set容器介绍及操作大全

《c++中的set容器介绍及操作大全》:本文主要介绍c++中的set容器介绍及操作大全,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录​​一、核心特性​​️ ​​二、基本操作​​​​1. 初始化与赋值​​​​2. 增删查操作​​​​3. 遍历方

解析C++11 static_assert及与Boost库的关联从入门到精通

《解析C++11static_assert及与Boost库的关联从入门到精通》static_assert是C++中强大的编译时验证工具,它能够在编译阶段拦截不符合预期的类型或值,增强代码的健壮性,通... 目录一、背景知识:传统断言方法的局限性1.1 assert宏1.2 #error指令1.3 第三方解决

C++11委托构造函数和继承构造函数的实现

《C++11委托构造函数和继承构造函数的实现》C++引入了委托构造函数和继承构造函数这两个重要的特性,本文主要介绍了C++11委托构造函数和继承构造函数的实现,具有一定的参考价值,感兴趣的可以了解一下... 目录引言一、委托构造函数1.1 委托构造函数的定义与作用1.2 委托构造函数的语法1.3 委托构造函

C++11作用域枚举(Scoped Enums)的实现示例

《C++11作用域枚举(ScopedEnums)的实现示例》枚举类型是一种非常实用的工具,C++11标准引入了作用域枚举,也称为强类型枚举,本文主要介绍了C++11作用域枚举(ScopedEnums... 目录一、引言二、传统枚举类型的局限性2.1 命名空间污染2.2 整型提升问题2.3 类型转换问题三、C