文档章节

【C++11】30分钟了解C++11新特性

王选易
 王选易
发布于 2013/12/10 23:33
字数 1574
阅读 30225
收藏 198

点击进入我的新博客


###什么是C++11

C++11是曾经被叫做C++0x,是对目前C++语言的扩展和修正,C++11不仅包含核心语言的新机能,而且扩展了C++的标准程序库**(STL),并入了大部分的C++ Technical Report 1(TR1)**程序库(数学的特殊函数除外)。

C++11包括大量的新特性:包括lambda表达式,类型推导关键字auto、decltype,和模板的大量改进。

本文将对C++11的以上新特性进行简单的讲解,以便大家能够快速了解到C++11对C++的易用性方面祈祷的巨大作用。

如果您觉得本文的排版不是很舒服,可以查看我的PDF文档:百度盘链接

###新的关键字

####auto

C++11中引入auto第一种作用是为了自动类型推导

auto的自动类型推导,用于从初始化表达式中推断出变量的数据类型。通过auto的自动类型推导,可以大大简化我们的编程工作

auto实际上实在编译时对变量进行了类型推导,所以不会对程序的运行效率造成不良影响

另外,似乎auto并不会影响编译速度,因为编译时本来也要右侧推导然后判断与左侧是否匹配。

<!-- lang: cpp -->
auto a; // 错误,auto是通过初始化表达式进行类型推导,如果没有初始化表达式,就无法确定a的类型
auto i = 1;
auto d = 1.0;
auto str = "Hello World";
auto ch = 'A';
auto func = less<int>();
vector<int> iv;
auto ite = iv.begin();
auto p = new foo() // 对自定义类型进行类型推导

auto不光有以上的应用,它在模板中也是大显身手,比如下例这个加工产品的例子中,如果不使用auto就必须声明Product这一模板参数:

template <typename Product, typename Creator>
void processProduct(const Creator& creator) {
    Product* val = creator.makeObject();
    // do somthing with val
}		
		. 

如果使用auto,则可以这样写:

template <typename Creator>
void processProduct(const Creator& creator) {
    auto val = creator.makeObject();
    // do somthing with val
}

抛弃了麻烦的模板参数,整个代码变得更加正解了。

####decltype

decltype实际上有点像auto的反函数,auto可以让你声明一个变量,而decltype则可以从一个变量或表达式中得到类型,有实例如下:

int x = 3;
decltype(x) y = x;

有人会问,decltype的实用之处在哪里呢,我们接着上边的例子继续说下去,如果上文中的加工产品的例子中我们想把产品作为返回值该怎么办呢?我们可以这样写:

template <typename Creator>
auto processProduct(const Creator& creator) -> decltype(creator.makeObject()) {
    auto val = creator.makeObject();
    // do somthing with val
}

####nullptr

nullptr是为了解决原来C++中NULL的二义性问题而引进的一种新的类型,因为NULL实际上代表的是0,

void F(int a){
    cout<<a<<endl;
}

void F(int *p){
    assert(p != NULL);
    
    cout<< p <<endl;
}

int main(){
    
    int *p = nullptr;
    int *q = NULL;
    bool equal = ( p == q ); // equal的值为true,说明p和q都是空指针
    int a = nullptr; // 编译失败,nullptr不能转型为int
    F(0); // 在C++98中编译失败,有二义性;在C++11中调用F(int)
    F(nullptr);

    return 0;
}

###序列for循环

在C++中for循环可以使用类似java的简化的for循环,可以用于遍历数组,容器,string以及由begin和end函数定义的序列(即有Iterator),示例代码如下:

map<string, int> m{{"a", 1}, {"b", 2}, {"c", 3}};
for (auto p : m){
    cout<<p.first<<" : "<<p.second<<endl;
}

###Lambda表达式

lambda表达式类似Javascript中的闭包,它可以用于创建并定义匿名的函数对象,以简化编程工作。Lambda的语法如下:

[函数对象参数](操作符重载函数参数)->返回值类型{函数体}

vector<int> iv{5, 4, 3, 2, 1};
int a = 2, b = 1;

for_each(iv.begin(), iv.end(), [b](int &x){cout<<(x + b)<<endl;}); // (1)

for_each(iv.begin(), iv.end(), [=](int &x){x *= (a + b);});		// (2)

for_each(iv.begin(), iv.end(), [=](int &x)->int{return x * (a + b);});// (3)
  • []内的参数指的是Lambda表达式可以取得的全局变量。(1)函数中的b就是指函数可以得到在Lambda表达式外的全局变量,如果在[]中传入=的话,即是可以取得所有的外部变量,如(2)和(3)Lambda表达式
  • ()内的参数是每次调用函数时传入的参数
  • ->后加上的是Lambda表达式返回值的类型,如(3)中返回了一个int类型的变量

###变长参数的模板

我们在C++中都用过pair,pair可以使用make_pair构造,构造一个包含两种不同类型的数据的容器。比如,如下代码:

auto p = make_pair(1, "C++ 11");

由于在C++11中引入了变长参数模板,所以发明了新的数据类型:tuple,tuple是一个N元组,可以传入1个, 2个甚至多个不同类型的数据

auto t1 = make_tuple(1, 2.0, "C++ 11");
auto t2 = make_tuple(1, 2.0, "C++ 11", {1, 0, 2});

这样就避免了从前的pair中嵌套pair的丑陋做法,使得代码更加整洁

另一个经常见到的例子是Print函数,在C语言中printf可以传入多个参数,在C++11中,我们可以用变长参数模板实现更简洁的Print

template<typename head, typename... tail>
void Print(Head head, typename... tail) {
	cout<< head <<endl;
	Print(tail...);
}

Print中可以传入多个不同种类的参数,如下:

Print(1, 1.0, "C++11");

###更加优雅的初始化方法

在引入C++11之前,只有数组能使用初始化列表,其他容器想要使用初始化列表,只能用以下方法:

int arr[3] = {1, 2, 3}
vector<int> v(arr, arr + 3);

在C++11中,我们可以使用以下语法来进行替换:

int arr[3]{1, 2, 3};
vector<int> iv{1, 2, 3};
map<int, string>{{1, "a"}, {2, "b"}};
string str{"Hello World"};

###然后呢...

如果你想了解更多C++11令人兴奋的新特性,我会向你推荐这两个博客:

胡健的C++11系列博文

ToWrting的C++11系列博文

C++11的编译器支持列表

© 著作权归作者所有

共有 人打赏支持
王选易
粉丝 89
博文 20
码字总数 20066
作品 3
南京
程序员
私信 提问
加载中

评论(42)

s
sw55555
F(0); // 在C++98中编译失败,有二义性;在C++11中调用F(int)
这个是错的啊,0是int型的字符常量,是会直接调用F(int)
码农不会耕田
码农不会耕田
感觉越来越复杂啦... 没有把事情变得更轻松...
王选易
王选易

引用来自“MarvinGuo”的评论

最难理解的是右值

因为比较难理解,所以暂时没讲,说一说比较容易且能提升变成效率的特性
王选易
王选易

引用来自“billjyc1_h4u”的评论

膜拜软院大神79

你怎么知道我是软院的。。。1
billjyc1_h4u
billjyc1_h4u
膜拜软院大神79
书一
书一

引用来自“红薯”的评论

引用来自“王选易”的评论

引用来自“红薯”的评论

引用来自“王选易”的评论

引用来自“红薯”的评论

引用来自“王选易”的评论

@红薯 ,辛辛苦苦谢了1000字,求博客推荐4

你没看到你的标题里已经有“荐”了吗?:)

老大,博客什么时候添加RSS订阅功能啊。。。1

博客从第一天起就有rss订阅

老大,分享什么时候加上人人网和QQ空间啊

@廖凯 什么时候啊?

还有人人网啊?还没灭?
红薯
红薯

引用来自“王选易”的评论

引用来自“红薯”的评论

引用来自“王选易”的评论

引用来自“红薯”的评论

引用来自“王选易”的评论

@红薯 ,辛辛苦苦谢了1000字,求博客推荐4

你没看到你的标题里已经有“荐”了吗?:)

老大,博客什么时候添加RSS订阅功能啊。。。1

博客从第一天起就有rss订阅

老大,分享什么时候加上人人网和QQ空间啊

@廖凯 什么时候啊?
王选易
王选易

引用来自“红薯”的评论

引用来自“王选易”的评论

引用来自“红薯”的评论

引用来自“王选易”的评论

@红薯 ,辛辛苦苦谢了1000字,求博客推荐4

你没看到你的标题里已经有“荐”了吗?:)

老大,博客什么时候添加RSS订阅功能啊。。。1

博客从第一天起就有rss订阅

老大,分享什么时候加上人人网和QQ空间啊
红薯
红薯

引用来自“王选易”的评论

引用来自“红薯”的评论

引用来自“王选易”的评论

@红薯 ,辛辛苦苦谢了1000字,求博客推荐4

你没看到你的标题里已经有“荐”了吗?:)

老大,博客什么时候添加RSS订阅功能啊。。。1

博客从第一天起就有rss订阅
caltrop
caltrop
mark
【C++11】lambda 表达式解析

C++11 新增了很多特性,lambda 表达式是其中之一,如果你想了解的 C++11 完整特性,建议去这里,这里,这里,还有这里看看。本文作为 5 月的最后一篇博客,将介绍 C++11 的 lambda 表达式。 ...

筱骏
2017/03/17
0
0
OpenWRT开发之——对C++11的支持

前言 在上篇文章中博主尝试了在OpenWrt上用C++写个简单的程序测试了一下,可行。 博主这两天又了解了C++11,里面的新特性非常令我兴奋。比如sharedptr, lambda, auto都是非常有用的特性。[点...

临峰不畏
2015/05/18
0
11
【C++11】显式转换操作符

隐式类型转换是C++的一个既好又坏的特性。它给人以方便,但可能造成一些十分隐晦的错误。 类型转换提供了一个类型向另一个类型的构造。 class X{public: operator int() const noexcept { re...

风华神使
2015/09/28
150
0
【C++11】C++11新增的便利算法[1] all_of、any_of 和 none_of

本文选自:祁宇. 深入应用C++11:代码优化与工程级应用. 北京:机械工业出版社,2015.5 有改动。 算法库新增了3个用于判断的算法 allof、anyof 和 none_of: 其中: allof 检查区间 [ First,...

风华神使
2016/07/23
33
0
Mac上关于shell使用Python3和C++11声明

1.使用Python3 Mac上的shell上自带的Python版本是2.7,当需要使用Python3时,下载安装好Python时,在shell上敲入Python发现却还是显示Python2.7,这是因为Mac上的shell默认使用系统自带的Pyt...

希希里之海
2017/03/13
0
0

没有更多内容

加载失败,请刷新页面

加载更多

Confluence 6 教程:在 Confluence 中导航

当你对 Confluence 有所了解后,你会发现 Confluence 使用起来非常简单。这个教程主要是针对你使用的 Confluence 界面进行一些说明,同时向你展示在那里可以进行一些通用的任务和操作。 空间...

honeymose
13分钟前
2
0
sed, awk 练习

1. sed打印某行到某行之间的内容 2. sed 转换大小写 将单词首字母转化大写 将所有小写转化大写 3. sed 在某一行最后面添加一个数字 4. 删除某行到最后一行 解析: {:a;N;$!ba;d} :a : 是...

Fc丶
今天
2
0
babel6升级到7,jest-babel报错:Requires Babel "^7.0.0-0", but was loaded with "6.26.3".

自从将前端环境更新到babel7,jest-babel之前是基于babel6的,执行时候就会报:Requires Babel "^7.0.0-0", but was loaded with "6.26.3". 很烦,因为连续帮好几台电脑修复这个问题,所以记...

曾建凯
今天
1
0
探索802.11ax

802.11ax承诺在真实条件下改善峰值性能和最差情况。 如何改善今天的Wi-Fi? 在决定如何改进当前版本以外的Wi-Fi时,802.11ac,IEEE和Wi-Fi联盟调查了Wi-Fi部署和行为,以确定更广泛使用的障碍...

linuxprobe16
今天
2
0
使用linux将64G的SDCARD格式化为FAT32

一、命令如下: sudo fdisk -lsudo mkfs.vfat /dev/sda -Isudo fdisk /dev/sda Welcome to fdisk (util-linux 2.29.2). Changes will remain in memory only, until you decide to wri......

mbzhong
今天
4
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部