C++11学习笔记(四)

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是解引用后的对象}

因害怕失败而不敢放手一搏,永远不会成功

C++11学习笔记(四)

相关文章:

你感兴趣的文章:

标签云: