模板类型推导、auto推导
effective modern c++ 果然是神书,干货满满,简单记录下。
item1 模板推倒
典型的模板函数
temlate<class T>
void fn(ParamType param)
要记住的东西 |
---|
在模板类型推导的时候,有引用特性的参数的引用特性会被忽略 |
在推导通用引用参数的时候,左值会被特殊处理 |
在推导按值传递的参数时候,const 和/或volatile 参数会被视为非const 和非volatile |
在模板类型推导的时候,参数如果是数组或者函数名称,他们会被退化成指针,除非是用在初始化引用类型 |
当ParamType 是指针或者引用时,
如果
expr
的类型是个引用,忽略引用的部分。然后利用
expr
的类型和ParamType
对比去判断T
的类型。
template<typename T>
void f(T& param); // param是一个引用类型 int x = ; // x是一个int
const int cx = x; // cx是一个const int
const int& rx = x; // rx是const int的引用 f(x); // T是int,param的类型时int& f(cx); // T是const int,
// param的类型是const int&
f(rx); // T是const int
// param的类型时const int&
当ParamType 是右值引用时,(注意到param是左值时的特殊性)
如果
expr
是一个左值,T
和ParamType
都会被推导成左值引用。这有些不同寻常。第一,这是模板类型T
被推导成一个引用的唯一情况。第二,尽管ParamType
利用右值引用的语法来进行推导,但是他最终推导出来的类型是左值引用。如果
expr
是一个右值,那么就执行“普通”的法则(第一种情况)
template<typename T>
void f(T&& param); // param现在是一个通用的引用 int x = ; // 和之前一样
const int cx = x; // 和之前一样
const int& rx = x; // 和之前一样 f(x); // x是左值,所以T是int&
// param的类型也是int& f(cx); // cx是左值,所以T是const int&
// param的类型也是const int& f(rx); // rx是左值,所以T是const int&
// param的类型也是const int& f(); // 27是右值,所以T是int
// 所以param的类型是int&&
当ParamType 既不是指针也不是引用时,按照pass by value 来
和之前一样,如果
expr
的类型是个引用,将会忽略引用的部分。如果在忽略
expr
的引用特性,expr
是个const
的,也要忽略掉const
。如果是volatile
,照样也要忽略掉(volatile
对象并不常见。它们常常被用在实现设备驱动上面。查看更多的细节,请参考条款40。)
int x = ; // 和之前一样
const int cx = x; // 和之前一样
const int& rx = x; // 和之前一样 f(x); // T和param的类型都是int f(cx); // T和param的类型也都是int f(rx); // T和param的类型还都是int
最后再来个大杀器
//数组的引用再理解下
//int a[]={1,2,3} --> a的类型是啥? int [3]
//int & rb --->rb的类型是啥? rb先与&结合,表明是个引用,然后接触int 表明是个int型的引用
//int (&ra)[]={1,2,3} -->ra的类型是啥? ra先与&结合,表明是个引用,然后接触 int [3] 是个int [3] 的引用
//那如果去掉ra的形参,如何写呢 int (&) [3] ,这是个类型,并且是int [3] 的引用 //计算数组长度 constexpr需要vs2015以上
template<class T, std::size_t SIZE>
constexpr std::size_t ConstArraySize(T(&)[SIZE])
{
return SIZE;
} template<class T, std::size_t SIZE>
std::size_t ArraySize(T(&)[SIZE])
{
return SIZE;
}
item2 auto 推导
能够理解item1的话,这个就好办了,这个和item1的推导几乎一模一样。
item1的关键有个T 和ParamType 需要推导,对于auto来说,T就是auto!这么说有点困难,我们例子来说明
auto x = 27; //T 是auto ParamType也是auto 即item1上的规则3,pass by value
const auto cx = x; //T 是auto ParamType是 const auto ,按照规则3
auto& rx = x; //T 是auto ParamType是 auto & 按照规则1
auto&& uref1 = x; // x是int并且是左值
// 所以uref1的类型是int& auto&& uref2 = cx; // cx是int并且是左值
// 所以uref2的类型是const int& auto&& uref3 = ; // 27是int并且是右值
// 所以uref3的类型是int&& const char name[] = // name的类型是const char[13]
"R. N. Briggs"; auto arr1 = name; // arr1的类型是const char* auto& arr2 = name; // arr2的类型是const char (&)[13] void someFunc(int, double); // someFunc是一个函数,类型是
// void (*)(int, double) auto& func2 = someFunc; // func1的类型是
// void (&)(int, double)
到此为止,我们都非常成功,接下来是auto 与模板推导不同的地方
auto x1 = ; // 类型时int,值是27 auto x2(); // 同上 auto x3 = { }; // 类型是std::intializer_list<int>
// 值是{ 27 } auto x4{ }; // 同上 auto x5 = { , , 3.0 }; // 错误! 不能讲T推导成
// std::intializer_list<T> auto x = { , , }; // x的类型是
// std::initializer_list<int> template<typename T> // 和x的声明等价的
void f(T param); // 模板 f({ , , }); // 错误的!没办法推导T的类型 template<typename T>
void f(std::initializer_list<T> initList); f({ , , }); // T被推导成int,initList的
// 类型是std::initializer_list<int>
要记住的东西 |
---|
auto 类型推导通常和模板类型推导类似,但是auto 类型推导假定花括号初始化代表的类型是std::initializer_list ,但是模板类型推导却不是这样 |
auto 在函数返回值或者lambda参数里面执行模板的类型推导,而不是通常意义的auto 类型推导 |
模板类型推导、auto推导的更多相关文章
- C++11 - 类型推导auto关键字
在C++11中,auto关键字被作为类型自动类型推导关键字 (1)基本用法 C++98:类型 变量名 = 初值; int i = 10; C++11:auto 变量名 = 初值; auto i ...
- Effective Modern C++翻译(7)-条款6:当auto推导出意外的类型时,使用显式的类型初始化语义
条款6:当auto推导出意外的类型时,使用显式的类型初始化语义 条款5解释了使用auto来声明变量比使用精确的类型声明多了了很多的技术优势,但有的时候,当你想要zag的时候,auto可能会推导出了zi ...
- C++ 11 学习1:类型自动推导 auto和decltype
Cocos 3.x 用了大量的C++ 11 的东西,所以作为一个C++忠实粉丝,有必要对C++ 11进行一个系统的学习. 使用C++11之前,一定要注意自己使用的编译器对C++11的支持情况,有些编译 ...
- Effective Modern C++翻译(2)-条款1:明白模板类型推导
第一章 类型推导 C++98有一套单一的类型推导的规则:用来推导函数模板,C++11轻微的修改了这些规则并且增加了两个,一个用于auto,一个用于decltype,接着C++14扩展了auto和dec ...
- item 6: 当auto推导出一个不想要的类型时,使用显式类型初始化的语法
本文翻译自<effective modern C++>,由于水平有限,故无法保证翻译完全正确,欢迎指出错误.谢谢! 博客已经迁移到这里啦 Item 5解释了比起显式指定类型,使用auto来 ...
- C++11 类型推导auto
在C++11之前,auto关键字用来指定存储期.在新标准中,它的功能变为类型推断.auto现在成了一个类型的占位符,通知编译器去根据初始化代码推断所声明变量的真实类型.使用auto会拖慢c++效率吗? ...
- C++11 自动推导auto
C++11 自动推导auto C++11中引入的auto主要有两种用途:自动类型推导和返回值占位. auto在C++98中的标识临时变量的语义,由于使用极少且多余,在C++11中已被删除.前后两个标准 ...
- C++隐式推导-auto关键词
总述 C++中有一个关键字,它不进行显式声明,而进行隐式推导,auto可以在声明变量时根据变量初始值的类型自动为此变量选择匹配的类型.C++语言类似的关键字还有decltype. 如何评价 C++ 1 ...
- 现代C++之理解模板类型推断(template type deduction)
理解模板类型推断(template type deduction) 我们往往不能理解一个复杂的系统是如何运作的,但是却知道这个系统能够做什么.C++的模板类型推断便是如此,把参数传递到模板函数往往能让 ...
随机推荐
- Linux再学习(一)-学习路线规划
1 抛弃旧文化,迎接Linux命令新文化 Linux第一步,从Windows思维,切换到Linux的"命令行+文件"模式 在Linux中,做什么都有相应命令.一般就在bin或者sb ...
- 从js 讲解时间复杂度和空间复杂度
1. 博客背景 今天有同事在检查代码的时候,由于函数写的性能不是很好,被打回去重构了,细思极恐,今天和大家分享一篇用js讲解的时间复杂度和空间复杂度的博客 2. 复杂度的表示方式 之前有看过的,你可能 ...
- 用mongodb 固定集合实现只保留固定数量的记录,自动淘汰老旧数据
在一个保存report记录的场景中,我们使用MongoDB进行数据存储 example: db: report Collection: daily_report 创建db: use report; ...
- win10+Anaconda3+CUDA9.0+CUDNN7.1+TensorFlow-gpu1.9+Pycharm
想在win10上运行下YOLO的例子,要先配置环境,折腾了两天,终于好了,整理下自己觉得有用且正确的流程. win10+Anaconda3+CUDA9.0+CUDNN7.1+TensorFlow1.9 ...
- ride.py在运行python3.×版本后导致无法运行及解决办法
最近一直在自学python自动化,网上看到rf框架挺适合初学自动化测试,于是通过虫师的搭建了rf框架, 但是在使用过程中遇到了一个问题,在网上没有找到明确解决办法于是想到记录一下 之前为了搭建rf框架 ...
- MQ如何解决消息的顺序性
一.消息的顺序性 1.延迟队列:设置一个全局变量index,根据实际情况一次按照index++的逻辑一次给消息队列设置延迟时间段,可以是0.5s,甚至1s; 弊端:如果A,B,C..消息队列消费时间不 ...
- JavaWeb——使用会话维持状态3
这次的例子是使用会话给上一个例子添加登陆功能 1.页面逻辑 首先是登陆页面,这里需要输入账号和密码,输入正确后将进入商品列表页面,输入错误将会提示账号或者密码错误 其次是商品列表和购物车页面,添加了注 ...
- 「求助」关于MacOS 适配不了SOIL的问题 以及我自己愚蠢的解决办法
我的环境 macOS High Sierra 10.13.6 (2018) 我的SOIL源是通过 终端 git clone https://github.com/DeVaukz/SOIL 直接从gay ...
- MTFlexbox自动化埋点探索
1. 背景 跨平台动态化技术是目前移动互联网领域的重点关注方向,它既能节约人力,又能实现业务快速上线的需求.经过十年的发展,美团App已经变成了一个承载众多业务的超级平台,众多的业务方对业务形态的快速 ...
- Go_笔试题记录-指针与值类型实现接口的区别
1.如果Add函数的调用代码为: func main() { var a Integer = 1 var b Integer = 2 var i interface{} = &a sum := ...