POD是 Plain Old Data的缩写,用于说明一个类型的属性。
C++11将POD 划分为2个基本概念的合集——平凡的 和 标准布局的。
“平凡的” 类或结构体 应该符合以下定义
1.平凡的 构造、析构函数;
2.平凡的复制、移动构造函数;
3.平凡的复制赋值运算符、移动赋值运算符;
4.不能包含 虚函数 以及 虚基类
可以通过一些辅助的类模板对以上属性进行判断。
template <typename T> struct std::is_trivial;下面看一个示例#include <iostream>#include <type_traits>using namespace std;struct Trivial1 {};struct Trivial2 {public:int a;private:int b;};struct Trivial3 {Trivial1 a;Trivial2 b;};struct Trivial4 {Trivial2 a[23];};struct Trivial5 {int x;static int y; };struct NonTrivial1 {NonTrivial1() : z(42) {}int z;};struct NonTrivial2 {NonTrivial2();int w;};NonTrivial2::NonTrivial2() = default;struct NonTrivial3 {Trivial5 c;virtual void f(); };int main() {cout << is_trivial<Trivial1>::value << endl; // 1cout << is_trivial<Trivial2>::value << endl; // 1cout << is_trivial<Trivial3>::value << endl; // 1cout << is_trivial<Trivial4>::value << endl; // 1cout << is_trivial<Trivial5>::value << endl; // 1cout << is_trivial<NonTrivial1>::value << endl; // 0cout << is_trivial<NonTrivial2>::value << endl; // 0cout << is_trivial<NonTrivial3>::value << endl; // 0return 0;}可以以此结果来对比 “平凡的” 这一特性。
POD 的另一个概念是 标准布局。这一部分 还不太了解,先不写了。
【模板的别名】
在C++中,我们习惯用typedef来定义类的别名,在C++11中,使用using同样可以定义类型的别名。
【基于范围的for循环】
在C++98标准下遍历一个数组
#include <iostream>using namespace std;int main() {int arr[5] = { 1, 2, 3, 4, 5};int * p;for (p = arr; p < arr + sizeof(arr)/sizeof(arr[0]); ++p){*p *= 2;}for (p = arr; p < arr + sizeof(arr)/sizeof(arr[0]); ++p){cout << *p << '\t';}}当然,也可以通过类模板来实现#include <algorithm>#include <iostream>using namespace std;int action1(int & e){ e *= 2; }int action2(int & e){ cout << e << '\t'; }int main() {int arr[5] = { 1, 2, 3, 4, 5};for_each(arr, arr + sizeof(arr)/sizeof(arr[0]), action1);for_each(arr, arr + sizeof(arr)/sizeof(arr[0]), action2);}最后来看下C++11中的实现#include <iostream>using namespace std;int main() {int arr[5] = { 1, 2, 3, 4, 5 };for (int & e: arr)e *= 2;for (int & e: arr)cout << e << '\t';}如果数组的大小不确定的话,是不能使用基于范围的for循环的#include <iostream>using namespace std;int func(int a[]) {for (auto e: a) // 编译失败cout << e;}int main() {int arr[] = {1, 2, 3, 4, 5};func(arr);}还有一点值得注意,当循环使用在标准库的容器中时, 如果使用auto来声明迭代的对象,那么,这个对象不会是迭代器对象#include <vector>#include <iostream>using namespace std;int main() {vector<int> v = {1, 2, 3, 4, 5};for (auto i = v.begin(); i != v.end(); ++i)cout << *i << endl;// i是迭代器对象for (auto e: v)cout << e << endl;// e是解引用后的对象}
因害怕失败而不敢放手一搏,永远不会成功