模板类型推导、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++的模板类型推断便是如此,把参数传递到模板函数往往能让 ...
随机推荐
- Redis分布式锁实战
什么是分布式锁 在单机部署的情况下,要想保证特定业务在顺序执行,通过JDK提供的synchronized关键字.Semaphore.ReentrantLock,或者我们也可以基于AQS定制化锁.单机部 ...
- 使用 Netty 实现一个 MVC 框架
NettyMVC 上面介绍 Netty 能做是什么时我们说过,相比于 SpringMVC 等框架,Netty 没提供路由等功能,这也契合和 Netty 的设计思路,它更贴近底层.下面我们在 Netty ...
- QT状态机
首先吐槽下网上各种博主不清不楚的讲解 特别容易让新手迷惑 总体思想是这样的:首先要有一个状态机对象, 顾名思义,这玩意就是用来容纳状态的.然后调用状态机的start()函数它就会更具你的逻辑去执行相关 ...
- java并发编程(十二)----(JUC原子类)数组类型介绍
上一节我们介绍过三个基本类型的原子类,这次我们来看一下数组类型: AtomicIntegerArray, AtomicLongArray, AtomicReferenceArray.其中前两个的使用方 ...
- node命令行工具之实现项目工程自动初始化的标准流程
一.目的 传统的前端项目初始流程一般是这样: 可以看出,传统的初始化步骤,花费的时间并不少.而且,人工操作的情况下,总有改漏的情况出现.这个缺点有时很致命. 甚至有马大哈,没有更新项目仓库地址,导致提 ...
- [转载]关于ActiveMQ集群
转载于 http://blog.csdn.net/nimmy/article/details/6247289 近日因工作关系,在研究JMS,使用ActiveMQ作为提供者,考虑到消息的重要,拟采用Ac ...
- xlistview错误
apply plugin: 'com.android.library' android { compileSdkVersion buildToolsVersion '26.0.1' defaultCo ...
- Winform改变Textbox边框颜色
using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; usin ...
- 100天搞定机器学习|day39 Tensorflow Keras手写数字识别
提示:建议先看day36-38的内容 TensorFlow™ 是一个采用数据流图(data flow graphs),用于数值计算的开源软件库.节点(Nodes)在图中表示数学操作,图中的线(edge ...
- JDK基础必备面试十问
1. new一个对象在Java内部做了哪些工作? 从静态角度来看,new一个对象表示创建一个类的对象实例. 从JVM运行角度来看,当JVM执行到new字节码时,首先会去查看类有没有被加载到内存以及初始 ...