C++ 11中引入了许多简化编程工作的语法上的新特性,我们暂且美其名曰“语法甜点”。下面一一进行介绍。
语法甜点1:序列for循环
序列for循环是一种简化的for循环,可用于遍历一组序列,包括各种容器、string、数组、初始化列表以及由begin和end函数定义的序列。示例代码如下:
1 vector<int> vctTemp{1, 2, 3};
2 for (auto a : vctTemp)
3 {
4 cout << a << endl;
5 }
复制代码
语法甜点2:委托构造函数
在引入C++ 11之前,如果某个类有多个重载的构造函数,且这些构造函数中有一些共同的初始化逻辑,通常都需要再编写一个带参数的初始化函数,然后在这些构造函数中调用这个初始化函数。在C++ 11中,再也不用这么麻烦了。我们可以实现一个最基础的构造函数,其他构造函数都调用这个构造函数。示例代码如下:
1 class CPerson
2 {
3 public:
4 CPerson() : CPerson(0, “”) { NULL; }
5 CPerson(int nAge) : CPerson(nAge, “”) { NULL; }
6 CPerson(int nAge, const string &strName)
7 {
8 stringstream ss;
9 ss << strName << “is ” << nAge << “years old.”;
10 m_strInfo = ss.str();
11 }
12
13 private:
14 string m_strInfo;
15 };
复制代码
语法甜点3:统一的初始化语法
在引入C++ 11之前,有各种不同的初始化语法。在C++ 11中,仍可以使用这些初始化语法,但也可以选择使用新引入的统一的初始化语法。统一的初始化语法用一对大括号{}表示,使用{}初始化语法还可有效地避免窄转换。示例代码如下:
1 int a{5};
2 char c{‘X’};
3 int p[5] = {1, 2,3, 4, 5};
4 vector<int> vctTemp{1, 2, 3};
5 CPerson person{10, “Mike”};
6 int b = 5.3; // b赋值成5,发生了窄转换
7 int d{5.3}; // 会提示编译错误,避免了窄转换
复制代码
语法甜点4:nullptr
nullptr是C++ 11中新加的一个关键字,用于标识空指针。引入nullptr后,可以解决某些函数重载时的二义性问题。示例代码如下:
1 void F(int a)
2 {
3 cout << a << endl;
4 }
5
6 void F(char *p)
7 {
8 assert(p != NULL);
9
10 cout << p << endl;
11 }
12
13 int main(int argc, _TCHAR* argv[])
14 {
15 int *p = nullptr;
16 int *q = NULL;
17 bool bEqual = (p == q); // 两个指针值是相等的,bEqual为true
18 int a = nullptr; // 编译失败,nullptr不是转换为int
19
20 F(0); // 在C++ 98中编译失败,有二义性;在C++ 11中调用F(int)
21 F(nullptr); // 调用F(char *)
22
23 getchar();
24 return 0;
25 }
复制代码
语法甜点5:成员变量初始化
与Java和C#中的用法一样,可以对成员变量进行就地初始化。示例代码如下:
1 class CPerson
2 {
3 private:
4 int m_nAge = 10;
5 string m_strName = “Mike”;
6 };
复制代码
语法甜点6:默认或禁用函数
当我们定义了自己的带参数的构造函数时,编译器将不再生成默认的构造函数,如果此时想使用默认的构造函数,则必须显式地声明并定义不带参数的构造函数。在C++ 11中,我们可以使用default关键字来表明我们希望使用默认的构造函数。类似的,当我们不想外部使用编译器自动生成的构造函数或赋值函数时,我们一般需要将其声明成protected或private的。在C++ 11中,我们可以使用delete关键字来表明我们不希望编译器生成默认的构造函数或赋值函数。示例代码如下:
1 class CPerson
2 {
3 public:
4 CPerson() = default;
5 CPerson(const CPerson &person) = delete;
6 };
复制代码
语法甜点7:继承的构造函数
当一个派生类的某个函数隐藏了基类中的某个同名函数时,如果我们想在派生类中导出基类中的这个同名函数,可以通过using Base::Func的方式将基类中的这个同名函数引入到派生类的作用域内。当该方法只对普通成员函数有效,不能用于构造函数。在C++ 11中,如果派生类认为基类的构造函数已经足够,则也可以使用using Base::Base的方式将基类的构造函数引入到派生类的作用域内。但需要注意的是,此时派生类中的成员变量并没有进行初始化,所以应当对这些成员变量进行就地初始化。示例代码如下:
1 class CBase
2 {
3 };
4
5 class CDerived : public CBase
6 {
7 public:
8 using CBase::CBase;
9 CDerived(int nData) : m_nData(nData) { NULL; }
10
11 private:
12 int m_nData = 10;
13 };
复制代码
语法甜点8:模板右边双括号
在C++ 98中,vector<vector<int>> vctTemp是一个非法的表达式,编译器会认为右边的>>是一个移位操作符,因此必须修改为vector<vector<int> > vctTemp,即在右边的两个>中间添加一个空格。在C++ 11中,这将不再是一个问题,编译器将能够识别出右边的双括号是两个模板参数列表的结尾。
语法甜点9:static_assert
静态断言static_assert由一个常量表达式和一个字符串构成。在编译期间,将计算常量表达式的值,如果为false,字符串将作为错误信息输出。示例代码如下:
1 char a = 10;
2 static_assert(sizeof(a)==4, “a is not an integer.”);
复制代码
语法甜点10:初始化列表
在引入C++ 11之前,只有数组能使用初始化列表。在C++ 11中,vector、list等各种容器以及string都可以使用初始化列表了。初始化列表对应的类为initializer_list,vector、list等各种容器以及string之所以可以使用初始化列表,是因为它们重载了参数类型为initializer_list的构造函数(称为初始化列表构造函数)和赋值函数(称为初始化列表赋值函数)。下面是一些使用初始化列表的例子。
1 void Print(const initializer_list<int> &ilData)
2 {
3 for (auto a : ilData)
4 {
5 cout << a << endl;
6 }
7 }
8
9 int main(int argc, _TCHAR* argv[])
10 {
11 vector<int> vctNum = {1, 2, 3, 4, 5};
12 map<string, string> mapID2Name = {{“92001″, “Jack”}, {“92002″, “Mike”}};
13 string strText{“hello world”};
14 Print({});
15 Print({1, 2});
16 Print({1, 2, 3, 4, 5});
17
18 getchar();
19 return 0;
20 }
复制代码
来源:http://www.cnblogs.com/hujian/archive/2012/02/20/2358853.html
>> 本文固定链接: http://www.vcgood.com/archives/3565