C++primer 13.3节练习

练习13.29

不会,因为该类中的数据成员都是内置类型的,而内置类型是没有特定版本的swap的,所以对swap的调用会调用标准库的std::swap;

练习13.30

 1 #include <iostream>
 2 #include <string>
 3 #include <memory>
 4
 5 using namespace std;
 6
 7
 8 class HasPtr {
 9     friend ostream &print(ostream &os, HasPtr &h);
10     friend void swap(HasPtr &lhs, HasPtr &rhs);
11 public:
12     HasPtr(const string &s = string()) : ps(new string(s)), i(0) {}
13     HasPtr(const HasPtr &ptr);
14     HasPtr &operator=(const HasPtr &pt);
15     ~HasPtr() { delete ps; }
16 private:
17     string *ps;
18     int i;
19 };
20
21 ostream &print(ostream &os, HasPtr &h);
22 inline void swap(HasPtr &lhs, HasPtr &rhs);
23
24 int main()
25 {
26     HasPtr has("hello");
27     HasPtr have = has;
28     print(cout, have);
29     HasPtr h1("world");
30     swap(have, h1);
31     print(cout, have);
32     print(cout, h1);
33     system("pause");
34     return 0;
35 }
36
37 HasPtr::HasPtr(const HasPtr & ptr) : ps(new string(*(ptr.ps))), i(ptr.i) {}
38
39 HasPtr & HasPtr::operator=(const HasPtr & pt)
40 {
41     ps = new string(*pt.ps);
42     i = pt.i;
43     return *this;
44     // TODO: 在此处插入 return 语句
45 }
46
47 ostream & print(ostream & os, HasPtr & h)
48 {
49     os << *h.ps << h.i << endl;
50     return os;
51     // TODO: 在此处插入 return 语句
52 }
53
54 inline void swap(HasPtr & lhs, HasPtr & rhs)
55 {
56     swap(lhs.ps, rhs.ps);
57     cout << "ps already copy" << endl;
58     swap(lhs.i, rhs.i);
59     cout << "i already copy" << endl;
60 }

练习13.31

 1 #include <iostream>
 2 #include <string>
 3 #include <memory>
 4 #include <vector>
 5 #include <algorithm>
 6 #include <numeric>
 7
 8 using namespace std;
 9
10 class HasPtr {
11     friend ostream &print(ostream &os, HasPtr &h);
12     friend void swap(HasPtr &lhs, HasPtr &rhs);
13     friend bool operator<(HasPtr &lpt, HasPtr &rpt);
14     friend string & out(HasPtr &pt);
15 public:
16     HasPtr(const string &s = string()) : ps(new string(s)), i(0) {}
17     HasPtr(const HasPtr &ptr);
18     HasPtr &operator=(const HasPtr &pt);
19     ~HasPtr() { delete ps; }
20
21 private:
22     string *ps;
23     int i;
24 };
25
26 ostream &print(ostream &os, HasPtr &h);
27 inline void swap(HasPtr &lhs, HasPtr &rhs);
28 bool operator<(HasPtr &lpt, HasPtr &rpt);
29 string & out(HasPtr &pt);
30
31 int main()
32 {
33     vector<HasPtr> svec;
34     string word;
35     while (cin >> word)
36     {
37         svec.push_back(*new HasPtr(word));
38     }
39     sort(svec.begin(), svec.end());
40     for (auto c : svec)
41     {
42         cout << out(c) << endl;
43     }
44     system("pause");
45     return 0;
46 }
47
48 HasPtr::HasPtr(const HasPtr & ptr) : ps(new string(*(ptr.ps))), i(ptr.i) {}
49
50 HasPtr & HasPtr::operator=(const HasPtr & pt)
51 {
52     auto q = new string(*pt.ps);
53     delete ps;
54     ps = q;
55     i = pt.i;
56     return *this;
57 }
58
59 string & out(HasPtr & pt)
60 {
61     return *pt.ps;
62 }
63
64 bool operator<(HasPtr & lpt, HasPtr &rpt)
65 {
66     return (*lpt.ps).size() < (*rpt.ps).size();
67 }
68
69 ostream & print(ostream & os, HasPtr & h)
70 {
71     os << *h.ps << h.i << endl;
72     return os;
73 }
74
75 inline void swap(HasPtr & lhs, HasPtr & rhs)
76 {
77     using std::swap;
78     swap(lhs.ps, rhs.ps);
79     cout << "ps already copy" << endl;
80     swap(lhs.i, rhs.i);
81     cout << "i already copy" << endl;
82 }

何时调用的swap函数不得而知。(疑问)

练习13.32

相比于类值版本的swap函数,指针版本交换的数据成员原本就是指针和整型,并没有优化的效果,所以应该没有得到益处。

时间: 2025-01-02 13:31:33

C++primer 13.3节练习的相关文章

C++primer 13.2节练习

练习13.22 1 #include <iostream> 2 #include <string> 3 #include <memory> 4 5 using namespace std; 6 7 8 class HasPtr { 9 friend ostream &print(ostream &os, HasPtr &h); 10 public: 11 HasPtr(const string &s = string()) : ps(s)

C++primer 13.5节练习

练习13.39 1 #include <iostream> 2 #include <string> 3 #include <utility> 4 #include <memory> 5 6 using namespace std; 7 8 class StrVec { 9 public: 10 StrVec() : elements(nullptr), first_free(nullptr), cap(nullptr) {} 11 StrVec(const

C++primer 13.1.3节练习

练习13.9 析构函数执行与构造函数相反的操作,构造函数初始化对象的非static数据成员,析构函数释放对象使用的资源,并销毁对象的非static数据成员.当一个类没有定义自己的析构函数的时候,编译器会为它定义一个合成析构函数. 练习13.10 在一个构造函数中,成员的初始化是在函数体执行之前完成的,且按照他们再类中出现的顺序进行初始化.再一个析构函数中,首先执行函数体,然后销毁成员.成员按照初始化顺序的逆序销毁. 练习13.11 1 #include <iostream> 2 #includ

C++primer 13.1.2节练习

练习13.6 其实就是"="运算,也就是赋值运算.右侧运算对象作为显示参数向左侧传递时时候.合成拷贝赋值运算符来禁止该类型对象的赋值.当一个类没有定义自己的拷贝赋值运算符的时候,编译器会为它自动生成一个合成拷贝赋值运算符. 练习13.7 将一个StrBlob赋值给另一个StrBlob这个操作是完全没有问题的. 当赋值StrBlobPtr的时候就会出错,在编译的时候就会报错,左右值的类型不一样(也不是继承关系),无法完成赋值. 练习13.8 1 #include <iostream

C++primer 13.6.2节练习

练习13.49 13.50 1 #include <iostream> 2 #include <string> 3 #include <utility> 4 #include <memory> 5 #include <algorithm> 6 #include <vector> 7 8 using namespace std; 9 10 class String { 11 friend ostream &print(ostre

C++primer 13.2.1节练习

练习13.23 1 #include <iostream> 2 #include <string> 3 #include <memory> 4 5 using namespace std; 6 7 8 class HasPtr { 9 friend ostream &print(ostream &os, HasPtr &h); 10 public: 11 HasPtr(const string &s = string()) : ps(ne

C++primer 13.6.3节练习

练习13.55 1 void push_back(const string& s) &&; 练习13.56 此时拷贝一个副本,但是问题来了,ret是一个左值,返回他的sorted函数,会不停的进行递归自己,而该函数并没有一个终止条件,所以最后堆栈会溢出,导致程序异常终止: 练习13.57 此时函数返回的是一个临时对象的sorted函数,而临时对象是一个右值,这时会调用右值的sorted重载函数,程序正常运行,但是原对象依然没有排序,因为他是一个const或者是一个左值,改变的只是临

C++primer 13.1.1节练习

练习13.1 如果一个构造函数的第一个参数是自身类类型的引用,且任何额外参数都有默认值,则此构造函数是拷贝构造函数:拷贝初始化通常使用拷贝构造函数来完成.拷贝构造函数被用来初始化非引用类类型参数: 练习13.2 拷贝构造函数自己的参数必须是引用类型.如果其参数不是引用类型,则调用永远也不会成功-为了调用拷贝构造函数,我们必须拷贝他的实参,但为了拷贝实参,我们又需要调用拷贝构造函数,如此无限循环: 练习13.3 如果我们没有定义自己的拷贝构造函数,则会调用编译器为我们合成的拷贝构造函数 练习13.

C++primer 13.1.6节练习

练习13.18 1 #include <iostream> 2 #include <string> 3 #include <memory> 4 5 using namespace std; 6 7 static int i = 0; 8 9 class Employee { 10 friend ostream& print(ostream& os, const Employee& emp); 11 public: 12 Employee() :