【C++ Primer 第13章】2. 拷贝控制和资源管理

拷贝控制和资源管理

• 类的行为像一个值。意味着它应该有自己的状态,当我们拷贝一个像值得对象时,副本和原对象是完全独立的,改变副本不会对原对象有任何影响。

• 行为像指针的类则共享状态。当我们拷贝一个这种类的对象时,副本和原对象使用相同的底层数据,改变副本也会改变原对象。

13.2节练习

 1 #include<iostream>
 2 #include<string>
 3 using namespace std;
 4 
 5 class HasPtr {
 6 public:
 7     HasPtr(const string &s = string()): ps(new string(s)), i(0) {}
 8     HasPtr(const HasPtr &rhs): ps(new string(*rhs.ps)), i(rhs.i) {}
 9 
10     HasPtr &operator=(const HasPtr &rhs);
11     HasPtr &operator=(const string &rhs);
12     string &operator*();
13 
14     ~HasPtr() { delete ps; }
15 private:
16     string *ps;
17     int i;
18 };
19 
20 HasPtr &HasPtr::operator=(const HasPtr &rhs)
21 {
22     auto newp = new string(*rhs.ps);
23     delete ps;
24     ps = newp;
25     i = rhs.i;
26     return *this;
27 }
28 
29 HasPtr &HasPtr::operator=(const string &rhs)
30 {
31     *ps = rhs;
32     return *this;
33 }
34 
35 string& HasPtr::operator*()
36 {
37     return *ps;
38 }
39 
40 int main()
41 {
42     HasPtr h("hi mom!");
43     HasPtr h2(h);
44     HasPtr h3(h);
45     h2 = "hi dad!";
46     h3 = "hi, son!";
47     cout << "h: " << *h << endl;
48     cout << "h2: " << *h2 << endl;
49     cout << "h3: " << *h3 << endl;
50     return 0;
51 }

运行结果:

 

 

行为像值的类

 1 #include<iostream>
 2 #include<string>
 3 using namespace std;
 4 
 5 class HasPtr {
 6 public:
 7     HasPtr(const string &s = string()): ps(new string(s)), i(0) {}
 8     HasPtr(const HasPtr &rhs): ps(new string(*rhs.ps)), i(rhs.i) {}
 9 
10     HasPtr& operator=(const HasPtr &rhs);
11     ~HasPtr() { delete ps; }
12 private:
13     string *ps;
14     int i;
15 };
16 
17 HasPtr& HasPtr::operator=(const HasPtr &rhs)
18 {
19     auto newp = new string(*rhs.ps);
20     delete ps;
21     ps = newp;
22     return *this;
23 }

 

 

定义行为像指针的类

 1 #include<iostream>
 2 #include<string>
 3 using namespace std;
 4 
 5 class HasPtr {
 6 public:
 7     HasPtr(const string &s = string()): ps(new string(s)), i(0), use(new size_t(1)) {}
 8     HasPtr(const HasPtr &rhs): ps(rhs.ps), i(rhs.i), use(rhs.use) { ++*use; }
 9 
10     HasPtr &operator=(const HasPtr &rhs);
11     ~HasPtr();
12 
13 private:
14     string * ps;
15     int i;
16     size_t *use;  //用来记录有多少个对象共享*ps的成员
17 };
18 
19 HasPtr &HasPtr::operator=(const HasPtr &rhs)
20 {
21     ++*rhs.use;
22     if (--*use == 0)
23     {
24         delete ps;
25         delete use;
26     }
27     ps = rhs.ps;
28     i = rhs.i;
29     use = rhs.use;
30     return *this;
31 }
32 
33 HasPtr::~HasPtr()
34 {
35     if (--*use == 0)
36     {
37         delete ps;
38         delete use;
39     }
40 }

 

posted @ 2018-05-07 19:26  苏格拉底的落泪  阅读(225)  评论(0编辑  收藏  举报