首页 > C/C++语言 > C/C++基本语法 > C++ 11语法甜点
2012
02-21

    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


留下一个回复