|
马上注册,结交更多好友,享用更多功能^_^
您需要 登录 才可以下载或查看,没有账号?立即注册
x
本帖最后由 拈花小仙 于 2014-7-9 00:12 编辑
$ i8 o* J5 S4 s$ u! ~1 o& p$ ~& O6 G0 b+ v# Q. B: h0 Q
什么是C++11
: f7 t2 @; J/ d5 TC++11是曾经被叫做C++0x,是对目前C++语言的扩展和修正,C++11不仅包含核心语言的新机能,而且扩展了C++的标准程序库(STL),并入了大部分的C++ Technical Report 1(TR1)程序库(数学的特殊函数除外)。
7 @) u- q+ \; [) D+ z: aC++11包括大量的新特性:包括lambda表达式,类型推导关键字auto、decltype,和模板的大量改进。
7 O' G% D" }* M$ T1 g本文将对C++11的以上新特性进行简单的讲解,以便大家能够快速了解到C++11对C++的易用性方面祈祷的巨大作用。 @* C l0 T9 R* k( I; [0 b
如果您觉得本文的排版不是很舒服,可以查看我的PDF文档:百度盘链接
! B- p5 o! N( c" v/ {4 N7 {+ |" _* w
新的关键字
: _) ^$ z2 j, r8 X; Q. q- n* X$ Z* u2 _- y5 V
auto# f; F+ B* T3 L! x) [
C++11中引入auto第一种作用是为了自动类型推导+ j' ?: [: M* D: {$ U8 b' }' S7 M
auto的自动类型推导,用于从初始化表达式中推断出变量的数据类型。通过auto的自动类型推导,可以大大简化我们的编程工作
. {9 y; N9 w- L4 `2 Zauto实际上实在编译时对变量进行了类型推导,所以不会对程序的运行效率造成不良影响
: I7 P+ V1 B M另外,似乎auto并不会影响编译速度,因为编译时本来也要右侧推导然后判断与左侧是否匹配。
0 [" c n4 ~4 T: F) s4 r" D2 G' k. E- C: D9 {& p9 c- V
& A9 `: w8 m0 w; K0 gauto a; // 错误,auto是通过初始化表达式进行类型推导,如果没有初始化表达式,就无法确定a的类型 - auto i = 1;
0 T7 w! o4 b4 @ r% P) s4 s, @0 M3 q% cauto d = 1.0; - auto str = "Hello World";
) |. D2 Y9 k1 n: y4 \( @' lauto ch = 'A'; - auto func = less<int>();
" }7 r, v2 G5 m! s7 Pvector<int> iv; - auto ite = iv.begin();
- auto p = new foo() // 对自定义类型进行类型推导
, a% a5 p9 a% o0 w8 \3 @ auto不光有以上的应用,它在模板中也是大显身手,比如下例这个加工产品的例子中,如果不使用auto就必须声明Product这一模板参数:% ]. G3 Q& o' l& F0 F
3 K5 Y6 \1 _* [2 U; g4 H1 u$ r" { k4 Q2 Q2 e
template <typename Product, typename Creator> - void processProduct(const Creator& creator) {
" L5 C0 S# d' s; n Product* val = creator.makeObject(); - // do somthing with val
- u( y( L8 {$ M( I} - .
0 X$ ~6 ~& j" u6 k- J 如果使用auto,则可以这样写:
3 @0 j& j( I$ _3 L0 q6 s: x% ] |) ^& F- n
" ~) o" R8 u$ ? V- gtemplate <typename Creator> - void processProduct(const Creator& creator) {
( y% S5 s6 F5 s) K. T" c5 I% [ auto val = creator.makeObject(); - // do somthing with val
- }
; q1 B6 a0 ^: m! j1 z 抛弃了麻烦的模板参数,整个代码变得更加正解了。
; G: p8 `- h3 Q$ H: M& H) ^! m# Y6 X; @8 F9 }
decltype
# Z- ]* x: @) x# |" N( Tdecltype实际上有点像auto的反函数,auto可以让你声明一个变量,而decltype则可以从一个变量或表达式中得到类型,有实例如下:
; @' H7 j x$ \1 _' ~% z* G' s1 {: b u' c
( T7 d1 n* N6 b6 iint x = 3; - decltype(x) y = x;4 i# k- r/ s* o9 n3 F/ ]
有人会问,decltype的实用之处在哪里呢,我们接着上边的例子继续说下去,如果上文中的加工产品的例子中我们想把产品作为返回值该怎么办呢?我们可以这样写:
+ a$ g) j! ]. [4 q; O
k; x& F/ v: [$ [& c
9 a& \ O8 f- W. a I4 X9 `$ W6 ttemplate <typename Creator> - auto processProduct(const Creator& creator) -> decltype(creator.makeObject()) {
- ?/ H# u' S/ \. g( F; f/ l auto val = creator.makeObject(); - // do somthing with val
- }
# n& ?' k# |" p n nullptr6 h$ m, v# y6 _: x/ z( ?4 [3 c
nullptr是为了解决原来C++中NULL的二义性问题而引进的一种新的类型,因为NULL实际上代表的是0,5 s6 ~, ~6 Q& {9 R
! X& s8 O |0 w; f5 b
# N9 ]9 q7 G. Q- {4 R" ^& ^
void F(int a){ - cout<<a<<endl;
; ]1 F0 y& z7 j; p- L ? W} -
; ~6 Y- f! a" D' r* }void F(int *p){ - assert(p != NULL);
- cout<< p <<endl;
) b% h, c+ H5 T! P% \/ D} -
. _ v+ a; ^9 b' x5 `; {, O. |8 qint main(){ -
$ ]& p1 `2 y6 Z# W& f int *p = nullptr; - int *q = NULL;
& d+ m# j7 L G bool equal = ( p == q ); // equal的值为true,说明p和q都是空指针 - int a = nullptr; // 编译失败,nullptr不能转型为int
# U2 I* z' J7 |3 f4 I/ Q, T F(0); // 在C++98中编译失败,有二义性;在C++11中调用F(int) - F(nullptr);
- return 0;
- }' A5 f9 Z! A$ m6 o( U4 `
序列for循环( t$ ?7 q% } n+ o( b( O
在C++中for循环可以使用类似java的简化的for循环,可以用于遍历数组,容器,string以及由begin和end函数定义的序列(即有Iterator),示例代码如下:
n4 y8 z+ Y7 y& m. E: e! E# ]) o+ v8 v2 ^1 Z3 I9 K6 _
8 ?( u( T' n7 a2 d+ X1 Vmap<string, int> m{{"a", 1}, {"b", 2}, {"c", 3}}; - for (auto p : m){
8 M: z! k2 i! p; I cout<<p.first<<" : "<<p.second<<endl; - }: m# c& [/ Y0 C, t5 x) ^+ y
Lambda表达式
( T- z& u$ y+ p# ~lambda表达式类似Javascript中的闭包,它可以用于创建并定义匿名的函数对象,以简化编程工作。Lambda的语法如下:
1 F# u; \; f& q' I* @+ I, j$ J& x[函数对象参数](操作符重载函数参数)->返回值类型{函数体}
. J; H9 _+ E; I+ d4 w
+ w# L/ X: x/ q' D+ x5 ^% B$ Q% C; q- |
vector<int> iv{5, 4, 3, 2, 1}; - int a = 2, b = 1;
- for_each(iv.begin(), iv.end(), (int &x){cout<<(x + b)<<endl;}); // (1)
- b) B8 i, I! s: o! m. E8 D; ?
* z) `" i( g6 E3 |6 s- 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)
+ [& E7 U7 n. J! w5 J: B% W [b]- []内的参数指的是Lambda表达式可以取得的全局变量。(1)函数中的b就是指函数可以得到在Lambda表达式外的全局变量,如果在[]中传入=的话,即是可以取得所有的外部变量,如(2)和(3)Lambda表达式
- ()内的参数是每次调用函数时传入的参数。
- ->后加上的是Lambda表达式返回值的类型,如(3)中返回了一个int类型的变量- ]: j6 W% p/ c; J4 b
, j3 h- T& x# J3 d, m
变长参数的模板
( k3 n+ W2 ?, i) y我们在C++中都用过pair,pair可以使用make_pair构造,构造一个包含两种不同类型的数据的容器。比如,如下代码:' y5 d( s2 K6 q7 ?" n1 W7 a8 Q
- auto p = make_pair(1, "C++ 11");7 R `4 L5 _( s) {0 _8 p
由于在C++11中引入了变长参数模板,所以发明了新的数据类型:tuple,tuple是一个N元组,可以传入1个, 2个甚至多个不同类型的数据
! s1 F0 e) [2 G1 i7 n
& V# @# _" {3 C2 u% Q! q
% @! _) I" E* h' W, N% w8 z( j6 pauto t1 = make_tuple(1, 2.0, "C++ 11"); - auto t2 = make_tuple(1, 2.0, "C++ 11", {1, 0, 2});
" X) H, S1 A) i; d2 P 这样就避免了从前的pair中嵌套pair的丑陋做法,使得代码更加整洁
, m- _! m# l% ~2 T; L另一个经常见到的例子是Print函数,在C语言中printf可以传入多个参数,在C++11中,我们可以用变长参数模板实现更简洁的Print
" o4 g" y/ [/ }0 }( ^) J% s" k# n( H9 l
8 A N0 ?% x, Y4 E- H8 h. L
template<typename head, typename... tail> - void Print(Head head, typename... tail) {
. J" @: Y2 S H1 T7 P; m, | cout<< head <<endl; - Print(tail...);
- }
* `; I w) X/ T) |4 a6 B: h Print中可以传入多个不同种类的参数,如下:
* H I! l: w# r+ r. W- Print(1, 1.0, "C++11");" ^4 b# m/ A8 h" T
更加优雅的初始化方法1 a7 O4 W' M5 @; Q. }: R
在引入C++11之前,只有数组能使用初始化列表,其他容器想要使用初始化列表,只能用以下方法:
# e: M# L$ c) w, @. X* P. ~% n& W( ?9 K
& @# I2 |" ^. p7 l+ G$ {. `
int arr[3] = {1, 2, 3} - vector<int> v(arr, arr + 3);
( Z. c$ p; U- N' x* z$ k 在C++11中,我们可以使用以下语法来进行替换:, O& r. Q. j1 X4 F
0 X: I8 E& f" d. i- ?8 `0 R# w+ k
- Z* l4 n5 h% C3 a3 Y
int arr[3]{1, 2, 3}; - vector<int> iv{1, 2, 3}; 6 [' z; P {! Z l: T
map<int, string>{{1, "a"}, {2, "b"}}; - string str{"Hello World"};
' }5 @) n! _( n9 u6 Y [/b] |
|