C++11 如此受欢迎,你都用过哪些新特性?(一)
随着 C++ 的现代化进程在加快,前段时间有些朋友和我提起,现在的 C++ 语法他/她已经看不懂了。有必要让初进 C++ 的朋友对 C++ 语言的现代化版本有更全面的认识,藉此机会,我们就一起聊聊现代化的起点版本 C++ 11 引入了哪些比较有用的新特性?
很多进入 C++ 领域的朋友都是从 C 转行而来,会天然地将 C 的语法用在 C++ 代码中。
随着 C++ 的现代化进程在加快,前段时间有些朋友和我提起,现在的 C++ 语法他/她已经看不懂了。
所以,有必要让初进 C++ 的朋友对 C++ 语言的现代化版本有更全面的认识,藉此机会,我们就一起聊聊现代化的起点版本 C++ 11 引入了哪些比较有用的新特性?
1. Lambda 表达式
Lambda 表达式其实就是无名函数,如果函数名字对于使用者来说没有意义,就很适合使用 Lambda 表达式替代。
比如往接口 interface() 传递回调函数时,以往一般采用传递函数地址,需要专门定义一个函数,特意为它起名还想了很久,而且自己从来不调用它,就很尴尬。
有了 Lambda 表达式,你可以:
interface([](int x, int y) { return x + y; });
2. enum class
之前我们使用的枚举 enum 有个非常让人懊恼的问题:
enum Color { Red, Blue, Green, Yellow };
enum TrafficLights { Red, Yellow, Green };
这样的定义会导致编译错误,因为枚举值的作用域是公共的,在 Color 中定义的枚举值 Red 会和 TrafficLights 中定义的 Red 之间有命名冲突。
C++ 11 引入了 enum class 强类型枚举,可以有效避免命名冲突,因为所有枚举值都被声明在特定的作用域内,引用时必须使用作用域解析符:
Color c = Color::Red;
除此之外,传统的枚举值可以被隐式转换成整数类型,而强类型枚举不能隐式转换,如需转换可用 static_cast:
int val = static_cast<int>(Color::Red);
3. 范围 for 循环
在之前的版本里,for 循环有很死板的写法,以遍历容器 vector 为例:
std::vector<int> v(5, 2);
for (std::vector<int>::iterator i = v.begin(); i != v.end(); ++ i) {
std::cout << *i << " ";
}
基本格式是:
for (初始化语句; 循环条件判断; 执行语句) {}
C++ 11 把 for 循环的格式扩展添加了对范围的遍历,改写上面的例子:
std::vector<int> v(5, 2);
for (int i : v) {
std::cout << i << " ";
}
范围 for 循环的写法简化了对容器、数组等类型数据的遍历。
4. 自动类型推导 auto
很多时候有些类型书写起来会很长,然后又要大量使用,想想都觉得头皮发麻。
就比如上面对容器执行迭代时,需要获取容器的迭代器,这个迭代器的类型是定义在容器内部的,声明容器的迭代器要书写的篇幅会很长,很不利于阅读:
std::vector<int>::iterator i;
所以,C++ 11 赋予了关键词 auto 新的功能,只要使用 auto 声明变量,编译器就可以自动推导变量的类型,减少代码的冗余。
比如下面的表达式中,编译器可以依据右侧的操作数推断左侧新建的变量类型:
auto x = 5; // int
auto y = 3.14; // double
5. 空指针 nullptr
传统的代码里,我们随处可见到空指针的引用,比如常用于表示空指针的宏定义 NULL:
#define NULL 0
或者
#define NULL (void*)0
无论定义成哪种,编译器可以隐式将其转换成整形或者任何类型的指针,这是类型不安全的。
如果类内重载了这样一个方法:
// 声明 function
function(int x);
function(char *x);
那么调用时传入 NULL,编译器应该选择哪个方法?
function(NULL);
NULL 的用意不明确,同时也会带来阅读困难。
所以,C++ 11 引入了关键词 nullptr 专门表示空指针,它不能隐式转换成整形,也就没有了语义不明的问题。
int* p = nullptr;
6. constexpr
以往的代码里,有时想定义一个长度由变量定义的数组:
int array[num];
但是,这样的代码在 C++ 11 之前,编译器是会报错的。
还有,如果定义了一个函数执行计算:
int square(int x) {
return x * x;
}
调用时如果输入的参数是常量,难道非得要在运行时才费时执行计算吗?
int area = square(9);
C++ 11 引入了新关键词 constexpr,表示常量表达式,目标是榨干编译器的性能,减轻运行时负担,达到性能优化。
使用 constexpr 修饰变量,该变量在编译期就确定了值,在所有引用位置都会直接嵌入代码中,因而可以用于数组的长度定义。而 const 修饰的变量允许在运行时才确定。
constexpr int array[num];
而使用 constexpr 修饰函数,在调用该函数并输入常量或者 constexpr 修饰的量时,编译器会尝试执行计算并返回结果,这样运行时又可省略计算负担而直接得到结果:
constexpr int square(int x) {
return x * x;
}
int area = square(9);
7. 新关键词 override
、final
、noexcept
override
作为修饰词放在函数声明的末尾,声明当前函数是父类虚函数的重写,促使编译器去检查父类中是否有这样的虚函数,如果没有则报错。
class Base {
public:
virtual void func() const { ... } // 基类虚函数
};
class Derived : public Base {
public:
void func() const override { ... }
};
final
作为修饰词放在类名或者函数声明的末尾,它的作用是防止类被进一步继承或虚函数被进一步重写。
class Base {
virtual void func() { ... }
};
class Derived1 : public Base {
virtual void func() final {
// Derived1::func() 是最后一个版本
// Derived1 的任何派生类中都不能再重写 func()
}
};
class Derived2 final : public Derived1 {
// Derived2 是最终类,不能再被继承
};
noexcept
作为修饰词放在函数声明的末尾,显式声明函数在任何情况下都不会抛出异常。
class MyClass {
public:
MyClass(MyClass&&) noexcept = default; // 确保移动构造不会抛异常
~MyClass() noexcept = default; // 确保析构不会抛异常
};
使用修饰符 noexcept 有什么好处吗?
标准库里有些操作,在发生错误时,要求能够回滚操作,这样就需要安全的操作,包括不会抛出异常的操作,所以声明为 noexcept 的函数是客观需要的。
另外,移动操作比拷贝操作要有效率,但标准库在无法确定安全性的情况下,会选择更慢但更安全的拷贝操作,所以声明为 noexcept 的函数也对性能优化有帮助。
如果被声明为 noexcept 的函数却抛出了异常,系统会调用 terminate() 终止进程。
8. 统一的初始化语法
在传统的写法里,为了创建对象,会经常使用小括号 () 语法初始化对象,目的是调用构造函数初始化对象,但是有时会与函数声明产生混肴:
struct S {
S(int, int) {}
};
S obj1(1, 2); // 构造一个 S 对象
S obj2(); // 这是函数声明,函数名为 obj2,返回类型为 S 的对象
这种初始化的方式有很大的弊端。
有时初始化赋值,又有可能从较大范围的值隐式转换为较小范围的值,也就是窄化转换,会引发潜在的错误:
int x = 3.14; // 允许,x 将变为 3
在 C++ 11 之前,花括号 {} 主要用于数组初始化、聚合类型(如结构体)的初始化,以及通过构造函数的初始化列表进行成员初始化。
C++ 11 引入了统一初始化(也称为列表初始化),使得花括号可以在各种情况下使用,简化了不同类型的初始化方式,例如对象、数组、类成员等,都可以使用花括号进行初始化。
更重要的是,统一初始化解决了上面遇到的那些典型问题。
S obj3{1, 2}; // 明确表示构造一个 S 对象
int y{3.14}; // 错误,编译器禁止窄化转换
9. 右值引用 &&
在传统的写法里,有对左值的引用操作,但是缺乏对右值的引用操作,右值引用能优化对资源的使用,所以 C++ 11 引入了右值引用符号,用 &&
表示。
比如定义类的移动构造函数时,需要将形参声明为右值引用:
MyClass(MyClass&& other) noexcept : data(std::move(other.data)) {
std::cout << "Move Constructor\n";
}
八戒之前有篇文章专门介绍过现代版 C++ 的右值引用,欢迎进入我主页查看更多精彩内容。
上面主要聊的是语法相关的新特性,还有标准库提供的新特性将放在下一篇文章中,敬请期待...