30分钟了解C++11新特性
2013-12-18 13:29
375 查看
摘要 本文将对C++11的以上新特性进行简单的讲解,以便大家能够快速了解到C++11对C++的易用性方面祈祷的巨大作用。
C++ C++11
目录[-]
什么是C++11
新的关键字
auto
decltype
nullptr
序列for循环
Lambda表达式
变长参数的模板
更加优雅的初始化方法
然后呢…
C++11包括大量的新特性:包括lambda表达式,类型推导关键字auto、decltype,和模板的大量改进。
本文将对C++11的以上新特性进行简单的讲解,以便大家能够快速了解到C++11对C++的易用性方面祈祷的巨大作用。
如果您觉得本文的排版不是很舒服,可以查看我的PDF文档:百度盘链接
auto的自动类型推导,用于从初始化表达式中推断出变量的数据类型。通过auto的自动类型推导,可以大大简化我们的编程工作
auto实际上实在编译时对变量进行了类型推导,所以不会对程序的运行效率造成不良影响
另外,似乎auto并不会影响编译速度,因为编译时本来也要右侧推导然后判断与左侧是否匹配。
auto不光有以上的应用,它在模板中也是大显身手,比如下例这个加工产品的例子中,如果不使用auto就必须声明Product这一模板参数:
如果使用auto,则可以这样写:
抛弃了麻烦的模板参数,整个代码变得更加正解了。
有人会问,decltype的实用之处在哪里呢,我们接着上边的例子继续说下去,如果上文中的加工产品的例子中我们想把产品作为返回值该怎么办呢?我们可以这样写:
[函数对象参数](操作符重载函数参数)->返回值类型{函数体}
[]内的参数指的是Lambda表达式可以取得的全局变量。(1)函数中的b就是指函数可以得到在Lambda表达式外的全局变量,如果在[]中传入=的话,即是可以取得所有的外部变量,如(2)和(3)Lambda表达式
()内的参数是每次调用函数时传入的参数。
->后加上的是Lambda表达式返回值的类型,如(3)中返回了一个int类型的变量
由于在C++11中引入了变长参数模板,所以发明了新的数据类型:tuple,tuple是一个N元组,可以传入1个, 2个甚至多个不同类型的数据
这样就避免了从前的pair中嵌套pair的丑陋做法,使得代码更加整洁
另一个经常见到的例子是Print函数,在C语言中printf可以传入多个参数,在C++11中,我们可以用变长参数模板实现更简洁的Print
Print中可以传入多个不同种类的参数,如下:
在C++11中,我们可以使用以下语法来进行替换:
胡健的C++11系列博文
ToWrting的C++11系列博文
C++11的编译器支持列表
C++ C++11
目录[-]
什么是C++11
新的关键字
auto
decltype
nullptr
序列for循环
Lambda表达式
变长参数的模板
更加优雅的初始化方法
然后呢…
什么是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并不会影响编译速度,因为编译时本来也要右侧推导然后判断与左侧是否匹配。
1 | auto a; // 错误,auto是通过初始化表达式进行类型推导,如果没有初始化表达式,就无法确定a的类型 |
2 | auto i = 1; |
3 | auto d = 1.0; |
4 | auto str = "Hello World" ; |
5 | auto ch = 'A' ; |
6 | auto func = less< int >(); |
7 | vector< int > iv; |
8 | auto ite = iv.begin(); |
9 | auto p = new foo() // 对自定义类型进行类型推导 |
1 | template <typename Product,typename Creator> |
2 | void processProduct( const Creator& creator) { |
3 | Product* val = creator.makeObject(); |
4 | // do somthing with val |
5 | } |
6 | . |
1 | template <typename Creator> |
2 | void processProduct( const Creator& creator) { |
3 | auto val = creator.makeObject(); |
4 | // do somthing with val |
5 | } |
decltype
decltype实际上有点像auto的反函数,auto可以让你声明一个变量,而decltype则可以从一个变量或表达式中得到类型,有实例如下:1 | int x = 3 ; |
2 | decltype(x) y = x; |
1 | template <typename Creator> |
2 | auto processProduct( const Creator& creator) -> decltype(creator.makeObject()) { |
3 | auto val = creator.makeObject(); |
4 | // do somthing with val |
5 | } |
nullptr
nullptr是为了解决原来C++中NULL的二义性问题而引进的一种新的类型,因为NULL实际上代表的是0,01 | void F( int a){ |
02 | cout<<a<<endl; |
03 | } |
04 |
05 | void F( int *p){ |
06 | assert (p != NULL); |
07 |
08 | cout<< p <<endl; |
09 | } |
10 |
11 | int main(){ |
12 |
13 | int *p = nullptr; |
14 | int *q = NULL; |
15 | bool equal = ( p == q ); // equal的值为true,说明p和q都是空指针 |
16 | int a = nullptr; // 编译失败,nullptr不能转型为int |
17 | F( 0 ); // 在C++98中编译失败,有二义性;在C++11中调用F(int) |
18 | F(nullptr); |
19 |
20 | return 0 ; |
21 | } |
序列for循环
在C++中for循环可以使用类似java的简化的for循环,可以用于遍历数组,容器,string以及由begin和end函数定义的序列(即有Iterator),示例代码如下:1 | map<string, int > m{{ "a" , 1 }, { "b" , 2 }, { "c" , 3 }}; |
2 | for (auto p : m){ |
3 | cout<<p.first<< " : " <<p.second<<endl; |
4 | } |
Lambda表达式
lambda表达式类似Javascript中的闭包,它可以用于创建并定义匿名的函数对象,以简化编程工作。Lambda的语法如下:[函数对象参数](操作符重载函数参数)->返回值类型{函数体}
1 | vector< int > iv{ 5 , 4 , 3 , 2 , 1 }; |
2 | int a = 2 , b = 1 ; |
3 |
4 | for_each(iv.begin(), int &x){cout<<(x // (1) |
5 |
6 | for_each(iv.begin(), int &x){x // (2) |
7 |
8 | for_each(iv.begin(), int &x)-> int { return x * (a + b);}); // (3) |
()内的参数是每次调用函数时传入的参数。
->后加上的是Lambda表达式返回值的类型,如(3)中返回了一个int类型的变量
变长参数的模板
我们在C++中都用过pair,pair可以使用make_pair构造,构造一个包含两种不同类型的数据的容器。比如,如下代码:1 | auto p =make_pair( 1 , "C++ 11" ); |
1 | auto t1 = make_tuple( 1 , 2.0 , "C++ 11" ); |
2 | auto t2 = make_tuple( 1 , 2.0 , "C++ 11" , { 1 , 0 , 2 }); |
另一个经常见到的例子是Print函数,在C语言中printf可以传入多个参数,在C++11中,我们可以用变长参数模板实现更简洁的Print
1 | template<typename |
2 | void Print(Head |
3 | cout<< head <<endl; |
4 | Print(tail...); |
5 | } |
1 | Print( 1 , 1.0 , "C++11" ); |
更加优雅的初始化方法
在引入C++11之前,只有数组能使用初始化列表,其他容器想要使用初始化列表,只能用以下方法:1 | int arr[ 3 ] = { 1 , 2 , 3 } |
2 | vector< int > 3 ); |
1 | int arr[ 3 ]{ 1 , 2 , 3 }; |
2 | vector< int > iv{ 1 , 2 , 3 }; |
3 | map< int , string>{{ 1 , "a" }, { 2 , "b" }}; |
4 | string str{ "Hello World" }; |
然后呢…
如果你想了解更多C++11令人兴奋的新特性,我会向你推荐这两个博客:胡健的C++11系列博文
ToWrting的C++11系列博文
C++11的编译器支持列表
相关文章推荐
- C++学习之路—继承与派生(一):基本概念与基类成员的访问属性
- C++中指定小数位数输出
- 『C/C++』G++的命名返回值优化 -- NRVO
- c语言const关键字与宏的区别
- c++对象内存模型【内存布局】
- C语言数字与字符串转换 atoi()函数、itoa()函数、sprintf()函数
- JNIEnv的使用在C和C++中的区别
- c++Builder xe5 编译输出路径
- C++中重定义、重写、重载的区别以及隐藏与覆盖的访问
- Effective C++ 第二版 45)幕后行为 46)编译链接和运行时错误 47)非局部静态对象初始化
- c语言资源
- C语言运算符优先级和口诀 (转)
- C语言 指针出参使用方法
- C语言数字与字符串转换 atoi()函数、itoa()函数、sprintf()函数
- 高斯消元法求矩阵的逆
- c++基础知识整理二
- 【C++】拷贝构造函数和赋值符函数
- 关键字:C++ 4种类型转换 cast
- ubuntu下Qtcreator编写c/c++程序无法调试的解决办法
- C++基础知识学习整理一