c++ primer中文版第五版作业第十三章

仓库地址

文章目录

      • 13.1
      • 13.2
      • 13.3
      • 13.4
      • 13.5
      • 13.6
      • 13.7
      • 13.8
      • 13.9
      • 13.10
      • 13.11
      • 13.12
      • 13.13
      • 13.14
      • 13.15
      • 13.16
      • 13.17
      • 13.18
      • 13.19
      • 13.20
      • 13.21
      • 13.22
      • 13.23
      • 13.24
      • 13.25
      • 13.26
      • 13.27
      • 13.28
      • 13.29
      • 13.30
      • 13.31
      • 13.32
      • 13.33
      • 13.34
      • 13.35
      • 13.36
      • 13.37
      • 13.38
      • 13.39
      • 13.40
      • 13.41
      • 13.42
      • 13.43
      • 13.44
      • 13.45
      • 13.46
      • 13.47
      • 13.48
      • 13.49
      • 13.50
      • 13.51
      • 13.52
      • 13.53
      • 13.54
      • 13.55
      • 13.56
      • 13.57
      • 13.58

13.1

 如果一个构造函数的第一个参数是自身类类型的引用,且任何额外参数都有默认值,则此构造函数是拷贝构造函数。拷贝构造函数用于类类型的拷贝初始化。在以下情形中会发生类类型的拷贝初始化:

  • 使用=定义变量时
  • 将一个对象作为实参传递给一个非引用类型的形参
  • 从一个返回类型为非引用类型的函数返回一个对象
  • 用花括号列表初始化一个数组中的元素或一个聚合类中的成员

13.2

 拷贝构造函数的第一个参数必须时自身类类型的引用,否则为了调用拷贝构造函数,我们必须拷贝它的实参,但为了拷贝实参,我们又需要调用拷贝构造函数,如此无线循环,调用就不会成功。

13.3

StrBlobStrBlobPtr都没有定义自己的拷贝构造函数,编译器会为它们合成拷贝构造函数,合成的拷贝构造函数会将其参数的成员逐个拷贝到正在创建的对象中。StrBlob只有一个shared_ptr类型的数据成员,调用其拷贝构造函数,其引用计数加一。而StrBlobPtr有一个weak_ptr类型的数据成员,调用其拷贝构造函数,引用计数不增加。其size_t类型的数据成员直接拷贝。

13.4

Point foo_bar(Point arg)为非引用类型的形参传值,及为非引用类型的返回类型返回对象。
Point local=arg使用=进行拷贝初始化。
*heap=local使用=进行拷贝初始化。
Point pa[4]={local,*heap}用花括号列表初始化一个数组中的元素

13.5

class HasPtr
{
public:HasPtr(const std::string &s=std::string()):ps(new std::string(s)),i(0) {}HasPtr(const HasPtr &OHasPtr):ps(new std::string(*(OHasPtr.ps))),i(OHasPtr.i) {}
private:std::string *ps;int i;
}

13.6

  • 拷贝赋值运算符本身是一个重载的赋值运算符,定义为类的成员函数,左侧运算对象绑定到隐含的this参数,右侧运算对象接受一个与其所在类相同类型的参数。通常返回一个指向其左侧运算对象的引用。
  • 类的对象赋值时使用其定义的拷贝赋值运算符。
  • 用来禁止该类型对象的赋值,或者将右侧运算对象的每个非static成员赋予左侧运算对象的对应成员。
  • 如果一个类未定义自己的拷贝赋值运算符,编译器会为它生成一个合成拷贝赋值运算符。

13.7

StrBlobStrBlobPtr都没有定义自己的拷贝赋值运算符,编译器会为它们生成一个合成拷贝赋值运算符。两个StrBlob类型对象之间赋值,则会将右侧对象的shared_ptr成员赋值给左侧对象的shared_ptr成员,左侧对象的shared_ptr成员引用计数减一,右侧加一。两个StrBlobPtr类型对象之间赋值,则会将右侧对象的weak_ptr成员赋值给左侧对象的weak_ptr成员,再将右侧对象的size_t成员赋值给左侧对象的size_t类型成员,并返回左侧对象的引用。

13.8

class HasPtr
{
public:HasPtr(const std::string &s=std::string()):ps(new std::string(s)),i(0) {}HasPtr(const HasPtr &OHasPtr):ps(new std::string(*(OHasPtr.ps))),i(OHasPtr.i) {}HasPtr & operator=(const HasPtr &rhs) {std::string *newps=new std::string(*rhs.ps);delete ps;ps=newps;i=rhs.i;return *this;}
private:std::string *ps;int i;
}

13.9

  • 析构函数用来释放对象使用的资源,并销毁对象得非static数据成员。
  • 合成析构函数要嘛被用来阻止该类型的对象被销毁,要嘛函数体为空,在函数体之后进行析构阶段销毁对象非static数据成员。
  • 当一个类未定义自己的析构函数时,编译器会为它定义一个合成析构函数

13.10

  • StrBlob类未定义自己的析构函数,所以编译器会为它定义一个合成析构函数,该函数的函数体为空。StrBlob类型对象销毁时,在执行完空的函数体后会进行析构阶段,这时会调用shared_ptr类型数据成员自己的析构函数。
  • StrBlobPtr类未定义自己的析构函数,所以编译器会为它定义一个合成的析构函数,该函数的函数体为空。StrBlobPtr类型对象销毁时,在执行完空的函数体后会进行析构阶段,这时curr成员为内置类型,什么也不做。再依次执行weak_ptrshared_ptr的析构函数。

13.11

class HasPtr
{
public:HasPtr(const std::string &s=std::string()):ps(new std::string(s)),i(0) {}HasPtr(const HasPtr &OHasPtr):ps(new std::string(*(OHasPtr.ps))),i(OHasPtr.i) {}HasPtr & operator=(const HasPtr &rhs) {std::string *newps=new std::string(*rhs.ps);delete ps;ps=newps;i=rhs.i;return *this;}~HasPtr() {delete ps;}
private:std::string *ps;int i;
}

13.12

 会发生3次析构函数的调用,代码块结束时四个局部变量都离开了其作用域,将会被销毁。但trans内置指针类型,销毁它时什么都不会做,自然也就不会调用Sales_data的析构函数。

13.13

#include <iostream>
#include <vector>
using std::cout;
using std::endl;
using std::vector;
class X
{
public:X() {std::cout<<"X()"<<std::endl;}X(const X &) {std::cout<<"X(const X &)"<<std::endl;}X & operator=(const X &) {std::cout<<"X=X"<<std::endl;return *this;}~X() {std::cout<<"~X()"<<std::endl;}
private:
};
void f1(X x) {}
void f2(X &x) {}
int main(void)
{cout << "局部变量:" << endl;X x;cout << endl;cout << "非引用参数传递:" << endl;f1(x);cout << endl;cout << "引用参数传递:" << endl;f2(x);cout << endl;cout << "动态分配:" << endl;X *px = new X;cout << endl;cout << "添加到容器中:" << endl;vector<X> vx;vx.push_back(x);cout << endl;cout << "释放动态分配对象:" << endl;delete px;cout << endl;cout << "拷贝初始化和赋值:" << endl;X y = x;y = x;cout << endl;cout << "程序结束!" << endl;return 0;
}

13.14

 代码将输出三个相同的序号,因为合成的拷贝构造函数简单的将序号复制,这三个对象的需要都将等于对象a创建时的序号。

#include <iostream>
unsigned idx=0;
class numbered
{
friend void f(numbered s);
public:numbered():mysn(idx++) {}
private:unsigned mysn=0;
};
using std::cout;
using std::endl;
void f(numbered s)
{cout<<s.mysn<<endl;
}
int main(void)
{numbered a,b=a,c=b;f(a);f(b);f(c);return 0;
}

13.15

 可以改变上一题中调用的输出结果,b=a以及c=b都将调用拷贝构造函数为对象生成唯一的序号。而在调用函数f的过程中,由于对象作为实参传递给非引用类型的形参,又会调用拷贝构造函数为函数的局部对象生成唯一的序号。

#include <iostream>
unsigned idx=0;
class numbered
{
friend void f(numbered s);
public:numbered():mysn(idx++) {}numbered(const numbered &ori_num):mysn(idx++) {}
private:unsigned mysn=0;
};
using std::cout;
using std::endl;
void f(numbered s)
{cout<<s.mysn<<endl;
}
int main(void)
{numbered a,b=a,c=b;f(a);f(b);f(c);return 0;
}

13.16

 会改变输出结果,在调用函数f时就不会再调用numbered类型对象的拷贝构造函数,输出的就是分别a、b、c的序号数据成员。

#include <iostream>
unsigned idx=0;
class numbered
{
friend void f(const numbered &s);
public:numbered():mysn(idx++) {}numbered(const numbered &ori_num):mysn(idx++) {}
private:unsigned mysn=0;
};
using std::cout;
using std::endl;
void f(const numbered &s)
{cout<<s.mysn<<endl;
}
int main(void)
{numbered a,b=a,c=b;f(a);f(b);f(c);return 0;
}

13.17

如上。

13.18

class Employee
{
public:Employee():unique_sn(sn++) {}Employee(const std::string &str):name(str),unique_sn(sn++) {}
private:std::string name;unsigned unique_sn;static unsigned sn;
};
unsigned Employee::sn=0;

13.19

Employee类需要自己的拷贝控制成员,因为合成的拷贝构造函数和拷贝赋值运算符只会直接拷贝雇员证号数据成员。

class Employee
{
public:Employee():unique_sn(sn++) {}Employee(const std::string &str):name(str),unique_sn(sn++) {}Employee(const Employee &rhs):name(rhs.name),unique_sn(sn++) {}Employee & operator=(const Employee &rhs) {name=rhs.name;return *this;}~Employee() {}
private:std::string name;unsigned unique_sn;static unsigned sn;
};
unsigned Employee::sn=0;

13.20

TextQueryQueryResult类都未定义自己的拷贝控制成员,所以它们都会使用合成的版本,在合成版本的拷贝控制成员中,这两个类的shared_ptr、map、string类型的数据成员都会调用它们自己的拷贝控制成员完成拷贝、赋值、销毁的工作。

13.21

TextQueryQueryResult类的数据成员都是标准库类型,它们都定义有设计良好的拷贝控制成员,用合成的拷贝控制成员即可方便的管理这两个类型。所以它们不需要定义自己版本的拷贝控制成员。

13.22

class HasPtr
{public:HasPtr(const std::string &s=std::string()):pts(new std::string(s)) {}HasPtr(const HasPtr &org):value(org.value),pts(new std::string(*org.pts)) {}HasPtr & operator=(const HasPtr &rhs) {std::string *tmp=new std::string(*rhs.pts);delete pts;pts=tmp;value=rhs.value;return *this;}~HasPtr() {delete pts;}private:int value=0;std::string *pts;
};

13.23

如上

13.24

 未定义析构函数则无法释放指针数据成员所管理的内存,造成内存泄漏。未定义拷贝构造函数,则会简单的复制指针数据成员,使生成的新对象与原对象的指针数据成员指向同一个string

13.25

 拷贝构造函数以及拷贝赋值运算符需要使用原对象的data成员所指向的vector初始化被构造以及被赋值对象的data成员,如此可以复制该vector而不是拷贝shared_ptr成员。因为StrBlob类中只有一个shared_ptr类的数据成员,它有设计良好的析构函数,所以合成的析构函数可以顺利的完成相应工作。

13.26

StrBlob.h

#ifndef STRBLOB_H
#define STRBLOB_H
#include <memory>
#include <vector>
#include <string>
#include <initializer_list>
#include <stdexcept>
class StrBlobPtr;
class StrBlob
{
friend class StrBlobPtr;
public:typedef std::vector<std::string>::size_type size_type;StrBlob();StrBlob(std::initializer_list<std::string> il);StrBlob(const StrBlob &);StrBlob & operator=(const StrBlob &);size_type size() const {return data->size();}bool empty() const {return data->empty();}void push_back(const std::string &t){data->push_back(t);}void pop_back();std::string &front();std::string &back();const std::string &front() const;const std::string &back() const;StrBlobPtr begin();StrBlobPtr begin() const;StrBlobPtr end();StrBlobPtr end() const;
private:std::shared_ptr<std::vector<std::string>> data;void check(size_type i,const std::string &msg) const;
};
class StrBlobPtr
{
public:StrBlobPtr():curr(0){}StrBlobPtr(StrBlob &a,std::size_t sz=0):wptr(a.data),curr(sz){}StrBlobPtr(const StrBlob &a,std::size_t sz=0):wptr(a.data),curr(sz){}std::string &deref() const;StrBlobPtr &incr();
private:std::shared_ptr<std::vector<std::string>> check(std::size_t,const std::string &) const;std::weak_ptr<std::vector<std::string>> wptr;std::size_t curr=0;
};
#endif

StrBlob.cpp

#include "StrBlob.h"
using namespace std;
StrBlobPtr StrBlob::begin()
{return StrBlobPtr(*this);
}
StrBlobPtr StrBlob::begin() const
{return StrBlobPtr(*this);
}
StrBlobPtr StrBlob::end()
{return StrBlobPtr(*this,data->size());
}
StrBlobPtr StrBlob::end() const
{return StrBlobPtr(*this,data->size());
}StrBlob::StrBlob():data(make_shared<vector<string>>()){}
StrBlob::StrBlob(initializer_list<string> il):data(make_shared<vector<string>>(il)){}
StrBlob::StrBlob(const StrBlob &org):data(make_shared<std::vector<std::string>>(*org.data)) {}
StrBlob & StrBlob::operator=(const StrBlob &rhs)
{data=make_shared<std::vector<std::string>>(*rhs.data);return *this;
}
inline void StrBlob::check(size_type i,const string &msg) const
{if(i>=data->size())throw out_of_range(msg);
}
string &StrBlob::front()
{check(0,"front on empty StrBlob");return data->front();
}
string &StrBlob::back()
{check(0,"back on empty StrBlob");return data->back();
}
const std::string &StrBlob::front() const
{check(0,"front on empty StrBlob");return data->front();
}
const std::string &StrBlob::back() const
{check(0,"back on empty StrBlob");return data->back();
}
void StrBlob::pop_back()
{check(0,"pop_back on empty StrBlob");data->pop_back();
}std::shared_ptr<std::vector<std::string>> StrBlobPtr::check(std::size_t i,const std::string &msg) const
{auto ret=wptr.lock();if(!ret)throw std::runtime_error("unbound StrBlobPtr");if(i>=ret->size())throw std::out_of_range(msg);return ret;
}
std::string & StrBlobPtr::deref() const
{auto p=check(curr,"dereference past end");return (*p)[curr];
}
StrBlobPtr & StrBlobPtr::incr()
{check(curr,"increment past end of StrBlobPtr");++curr;return *this;
}

13.27

class HasPtr
{public:HasPtr(const std::string &s=std::string()):pts(new std::string(s)),value(0),use(new std::size_t(1)) {}HasPtr(const HasPtr &org):pts(org.pts),value(org.value),use(org.use) {++*use;}HasPtr & operator=(const HasPtr &rhs);~HasPtr();private:std::string *pts;int value;std::size_t *use;
};
HasPtr & HasPtr::operator=(const HasPtr &rhs)
{++*rhs.use;if(--*use==0){delete pts;delete use;}pts=rhs.pts;value=rhs.value;use=rhs.use;return *this;
}
HasPtr::~HasPtr()
{if(--*use){delete pts;delete use;}
}

13.28

class TreeNode
{public:TreeNode():count(1),left(nullptr),right(nullptr) {}TreeNode(const std::string &str=std::string(),TreeNode *lchild=nullptr,TreeNode *rchild=nullptr):value(str),count(1),left(lchild),right(rchild) {}void CopyTree();int ReleaseTree();TreeNode(const TreeNode &);~TreeNode();private:std::string value;int count;TreeNode *left;TreeNode *right;
};
class BinStrTree
{public:BinStrTree():root(nullptr) {}BinStrTree(TreeNode *tree=nullptr):root(tree) {}BinStrTree(const BinStrTree &);~BinStrTree();private:TreeNode *root;
};
BinStrTree::BinStrTree(const BinStrTree &org):root(org.root)
{root->CopyTree();
}
BinStrTree::~BinStrTree()
{if(!root->ReleaseTree())delete root;
}void TreeNode::CopyTree()
{if(left)left->CopyTree();if(right)right->CopyTree();++count;
}
int TreeNode::ReleaseTree()
{if(left){if(!left->ReleaseTree())delete left;}if(right){if(!right->ReleaseTree())delete  right;}--count;return count;
}
TreeNode::TreeNode(const TreeNode &org):value(org.value),count(1),left(org.left),right(org.right)
{if(left)left->CopyTree();if(right)right->CopyTree();
}
TreeNode::~TreeNode()
{if(count)ReleaseTree();
}

13.29

swap(HasPtr &,HasPtr &)中所调用的swap是属于标准库的std::swap版本,所以不会导致递归。

13.30

#include <iostream>
#include <string>
class HasPtr
{friend void swap(HasPtr &,HasPtr &);friend void print(const HasPtr &);public:HasPtr(const std::string &s=std::string()):pts(new std::string(s)) {}HasPtr(const HasPtr &org):value(org.value),pts(new std::string(*org.pts)) {}HasPtr & operator=(const HasPtr &rhs) {std::string *tmp=new std::string(*rhs.pts);delete pts;pts=tmp;value=rhs.value;return *this;}~HasPtr() {delete pts;}private:int value=0;std::string *pts;
};
void swap(HasPtr &lhs,HasPtr &rhs)
{using std::swap;swap(lhs.value,rhs.value);swap(lhs.pts,rhs.pts);std::cout<<"swap complete."<<std::endl;
}
void print(const HasPtr &hp)
{std::cout<<*hp.pts<<std::endl;
}
int main(void)
{HasPtr a("s1"),b("s2");swap(a,b);std::cout<<"a"<<std::endl;print(a);std::cout<<"b"<<std::endl;print(b);return 0;
}

13.31

#include <iostream>
#include <string>
#include <vector>
#include <algorithm>
class HasPtr
{friend void swap(HasPtr &,HasPtr &);friend std::ostream & print(std::ostream &,const HasPtr &);public:HasPtr(const std::string &s=std::string()):pts(new std::string(s)) {}HasPtr(const HasPtr &org):value(org.value),pts(new std::string(*org.pts)) {}HasPtr & operator=(const HasPtr &rhs) {std::string *tmp=new std::string(*rhs.pts);delete pts;pts=tmp;value=rhs.value;return *this;}bool operator<(const HasPtr &rhs) const {return *pts<*rhs.pts;}~HasPtr() {delete pts;}private:int value=0;std::string *pts;
};
inline void swap(HasPtr &lhs,HasPtr &rhs)
{using std::swap;swap(lhs.value,rhs.value);swap(lhs.pts,rhs.pts);std::cout<<"swap complete."<<std::endl;
}
std::ostream & print(std::ostream &os,const HasPtr &hp)
{os<<*hp.pts<<std::endl;return os;
}
using std::vector;
using std::cout;
using std::string;
int main(void)
{vector<HasPtr> hpv;string s="z y x a c d e f i h j k o q p r t s n m l";for(auto str:s)hpv.push_back(HasPtr(string("")+str));for(auto pv:hpv)print(cout,pv);sort(hpv.begin(),hpv.end());for(auto pv:hpv)print(cout,pv);return 0;
}

13.32

 类指针的HasPtr版本不会分配新的string副本,所以不需要自己的swap函数。

13.33

 因为首先我们要将Folder的指针存入Message的folders数据成员中,那么形参就不能是Folder,而应该是Folder的引用或指针。而save和remove函数中调用了Folder的addMsg和remMsg成员函数,这将改变Folder的数据成员,所以不能是const的。

13.34

Message.h

#ifndef _MESSAGE_H
#define _MESSAGE_H
#include <string>
#include <set>
class Folder;
class Message
{friend class Folder;friend void swap(Message &,Message &);public:explicit Message(const std::string &str=""):contents(str) {}Message(const Message &);Message & operator=(const Message &);~Message();void save(Folder &);void remove(Folder &);private:std::string contents;std::set<Folder *> folders;void add_to_Folders(const Message &);void remove_from_Folders();void addfolder(Folder *f) {folders.insert(f);}void remfolder(Folder *f) {folders.erase(f);}
};
class Folder
{friend void swap(Message &,Message &);friend class Message;public:Folder()=default;Folder(const Folder &f):messages(f.messages) {add_to_Messages(f);}Folder & operator=(const Folder &);~Folder();void save(Message &);void remove(Message &);private:std::set<Message *> messages;void addMsg(Message *m) {messages.insert(m);}void remMsg(Message *m) {messages.erase(m);}void add_to_Messages(const Folder &);void remove_from_Messages();
};
#endif

Message.cpp

#include "Message.h"
void Message::save(Folder &f)
{folders.insert(&f);f.addMsg(this);
}
void Message::remove(Folder &f)
{folders.erase(&f);f.remMsg(this);
}
void Message::add_to_Folders(const Message &m)
{for(auto f:m.folders)f->addMsg(this);
}
void Message::remove_from_Folders()
{for(auto f:folders)f->remMsg(this);
}
Message::Message(const Message &m):contents(m.contents),folders(m.folders)
{add_to_Folders(m);
}
Message & Message::operator=(const Message &rhs)
{remove_from_Folders();contents=rhs.contents;folders=rhs.folders;add_to_Folders(rhs);return *this;
}
Message::~Message()
{remove_from_Folders();
}
void swap(Message &lhs,Message &rhs)
{using std::swap;for(auto f:lhs.folders)f->remMsg(&lhs);for(auto f:rhs.folders)f->remMsg(&rhs);swap(lhs.contents,rhs.contents);swap(lhs.folders,rhs.folders);for(auto f:lhs.folders)f->addMsg(&lhs);for(auto f:rhs.folders)f->addMsg(&rhs);
}void Folder::add_to_Messages(const Folder &f)
{for(auto m:f.messages)m->addfolder(this);
}
void Folder::remove_from_Messages()
{for(auto m:messages)m->folders.erase(this);
}
void Folder::save(Message &m)
{messages.insert(&m);m.addfolder(this);
}
void Folder::remove(Message &m)
{messages.erase(&m);m.remfolder(this);
}
Folder & Folder::operator=(const Folder &rhs)
{remove_from_Messages();messages=rhs.messages;add_to_Messages(rhs);return *this;
}
Folder::~Folder()
{remove_from_Messages();
}

13.35

 如果使用合成的拷贝构造函数,那么新的Message对象不会被正确的放到对应的Folder中。如果使用合成的拷贝赋值运算符,那么左侧运算对象无法正确从Folder中删除,赋值后也无法放入对应的Folder中。合成的析构函数同样无法正确从Folder中删除。

13.36

见13.34

13.37

见13.34

13.38

 使用拷贝并交换的方式来设计Message的赋值运算符的话,会调用其拷贝构造函数及swap进行交换,这样会多次发生从所有目录中删除和添加到所有目录中的操作,效率低下。

13.39

StrVec.h

#ifndef _STRVEC_H
#define _STRVEC_H
#include <string>
#include <cstddef>
#include <utility>
#include <memory>
#include <initializer_list>
class StrVec
{public:StrVec():elements(nullptr),first_free(nullptr),cap(nullptr) {}StrVec(const StrVec &);StrVec(std::initializer_list<std::string>);StrVec & operator=(const StrVec &);~StrVec() {free();}void push_back(const std::string &);size_t size() {return first_free-elements;}size_t capacity() {return cap-elements;}std::string *begin() const {return elements;}std::string *end() const {return first_free;}void resize(const size_t n,const std::string &orgv=std::string());void reserve(const size_t n);private:std::string *elements;std::string *first_free;std::string *cap;static std::allocator<std::string> alloc;void chk_n_alloc() {if(size()==capacity()) reallocate();}std::pair<std::string *,std::string *> alloc_n_copy(const std::string *,const std::string *);void free();void reallocate();
};
#endif

StrVec.cpp

#include "StrVec.h"
StrVec::StrVec(const StrVec &org)
{std::pair<std::string *,std::string *> data=alloc_n_copy(org.begin(),org.end());elements=data.first;first_free=cap=data.second;
}
StrVec::StrVec(std::initializer_list<std::string> ls)
{auto data=alloc_n_copy(ls.begin(),ls.end());elements=data.first;first_free=cap=data.second;
}
StrVec & StrVec::operator=(const StrVec &rhs)
{auto data=alloc_n_copy(rhs.begin(),rhs.end());free();elements=data.first;first_free=cap=data.second;return *this;
}
void StrVec::push_back(const std::string &s)
{chk_n_alloc();alloc.construct(first_free++,s);
}
std::pair<std::string *,std::string *> StrVec::alloc_n_copy(const std::string *b,const std::string *e)
{std::string *data=alloc.allocate(e-b);return {data,uninitialized_copy(b,e,data)};
}
void StrVec::free()
{if(elements){for(auto p=first_free;p!=elements;)alloc.destroy(--p);alloc.deallocate(elements,cap-elements);}
}
void StrVec::reallocate()
{size_t newcapacity=size()?2*size():1;std::string *newdata=alloc.allocate(newcapacity);std::string *dest=newdata;std::string *src=elements;for(size_t i=0;i!=size();++i)alloc.construct(dest++,std::move(*src++));free();elements=newdata;first_free=dest;cap=elements+newcapacity;
}
void StrVec::reserve(const size_t n)
{if(n>capacity()){std::string *newdata=alloc.allocate(n);std::string *dest=newdata;std::string *src=elements;for(size_t i=0;i!=size();++i)alloc.construct(dest++,std::move(*src++));free();elements=newdata;first_free=dest;cap=elements+n;}
}
void StrVec::resize(const size_t n,const std::string &orgv)
{if(n<=size()){std::string *b=elements+n;std::string *e=first_free;while(b!=e)alloc.destroy(b++);first_free=elements+n;}else if(n<=capacity()){std::string *b=first_free;std::string *e=elements+n;while(b!=e)alloc.construct(b++,orgv);first_free=e;}else{reserve(n);std::string *b=first_free;std::string *e=elements+n;while(b!=e)alloc.construct(b++,orgv);first_free=e;}
}

13.40

见13.39

13.41

 使用后置递增运算符,那么相当于现在指针所指位置构造对象,再使指针前进。如果使用前置递增运算的话,first_free当前所指的位置就无法构造到对象。

13.42

13-42.cpp

#include <iostream>
#include <fstream>
#include "TextQuery.h"
#include "QueryResult.h"
using namespace std;
void runQueries(ifstream &infile)
{TextQuery tq(infile);while(true){cout<<"enter word to look for, or q to quit:";string s;if(!(cin>>s)||s=="q")break;print(cout,tq.query(s))<<endl;}
}
int main(int argc,char *argv[])
{ifstream infile(argv[1]);runQueries(infile);return 0;
}

TextQuery.h

#ifndef TEXT_QUERY_H
#define TEXT_QUERY_H
#include <string>
#include <vector>
#include <map>
#include <set>
#include <memory>
#include <fstream>
#include "QueryResult.h"
#include "StrVec.h"
class TextQuery
{
public:using line_no=std::vector<std::string>::size_type;TextQuery(std::ifstream &);QueryResult query(const std::string &) const;
private:std::shared_ptr<StrVec> file;std::map<std::string,std::shared_ptr<std::set<line_no>>> wm;
};
#endif

TextQuery.cpp

#include <fstream>
#include <iostream>
#include <sstream>
#include <vector>
#include <set>
#include <string>
#include <memory>
#include "TextQuery.h"
using namespace std;
TextQuery::TextQuery(ifstream &infile):file(new StrVec)
{string text;while(getline(infile,text)){file->push_back(text);size_t line_number=file->size()-1;istringstream line(text);string word;while(line>>word){shared_ptr<set<line_no>> &lines=wm[word];if(!lines)lines.reset(new set<line_no>);lines->insert(line_number);}}
}
QueryResult TextQuery::query(const string &sought) const
{static shared_ptr<set<line_no>> nodata(new set<line_no>);map<string,shared_ptr<set<line_no>>>::const_iterator map_it=wm.find(sought);if(map_it==wm.end())return QueryResult(sought,nodata,file);elsereturn QueryResult(sought,map_it->second,file);
}

QueryResult.h

#ifndef QUERYRESULT_H
#define QUERYRESULT_H
#include <iostream>
#include <string>
#include <memory>
#include <vector>
#include <set>
#include "StrVec.h"
class QueryResult
{
friend std::ostream &print(std::ostream &,const QueryResult &);
public:QueryResult(std::string s,std::shared_ptr<std::set<std::vector<std::string>::size_type>> l,std::shared_ptr<StrVec> f):sought(s),lines(l),file(f){}
private:std::string sought;std::shared_ptr<std::set<std::vector<std::string>::size_type>> lines;std::shared_ptr<StrVec> file;
};
std::ostream &print(std::ostream &,const QueryResult &);
inline std::string make_plural(std::size_t count,const std::string &word,const std::string &ending)
{return (count>1)?word+ending:word;
}
#endif

QueryResult.cpp

#include <iostream>
#include "QueryResult.h"
std::ostream &print(std::ostream &os,const QueryResult &qr)
{os<<qr.sought<<" occurs "<<qr.lines->size()<<" "<<make_plural(qr.lines->size(),"time","s")<<std::endl;for(auto num:*(qr.lines))os<<"\t(line "<<num+1<<")"<<*(qr.file->begin()+num)<<std::endl;return os;
}

StrVec.h

#ifndef _STRVEC_H
#define _STRVEC_H
#include <string>
#include <cstddef>
#include <utility>
#include <memory>
#include <initializer_list>
class StrVec
{public:StrVec():elements(nullptr),first_free(nullptr),cap(nullptr) {}StrVec(const StrVec &);StrVec(std::initializer_list<std::string>);StrVec & operator=(const StrVec &);~StrVec() {free();}void push_back(const std::string &);size_t size() {return first_free-elements;}size_t capacity() {return cap-elements;}std::string *begin() const {return elements;}std::string *end() const {return first_free;}void resize(const size_t n,const std::string &orgv=std::string());void reserve(const size_t n);private:std::string *elements;std::string *first_free;std::string *cap;static std::allocator<std::string> alloc;void chk_n_alloc() {if(size()==capacity()) reallocate();}std::pair<std::string *,std::string *> alloc_n_copy(const std::string *,const std::string *);void free();void reallocate();
};
#endif

StrVec.cpp

#include "StrVec.h"
std::allocator<std::string> StrVec::alloc;
StrVec::StrVec(const StrVec &org)
{std::pair<std::string *,std::string *> data=alloc_n_copy(org.begin(),org.end());elements=data.first;first_free=cap=data.second;
}
StrVec::StrVec(std::initializer_list<std::string> ls)
{auto data=alloc_n_copy(ls.begin(),ls.end());elements=data.first;first_free=cap=data.second;
}
StrVec & StrVec::operator=(const StrVec &rhs)
{auto data=alloc_n_copy(rhs.begin(),rhs.end());free();elements=data.first;first_free=cap=data.second;return *this;
}
void StrVec::push_back(const std::string &s)
{chk_n_alloc();alloc.construct(first_free++,s);
}
std::pair<std::string *,std::string *> StrVec::alloc_n_copy(const std::string *b,const std::string *e)
{std::string *data=alloc.allocate(e-b);return {data,uninitialized_copy(b,e,data)};
}
void StrVec::free()
{if(elements){for(auto p=first_free;p!=elements;)alloc.destroy(--p);alloc.deallocate(elements,cap-elements);}
}
void StrVec::reallocate()
{size_t newcapacity=size()?2*size():1;std::string *newdata=alloc.allocate(newcapacity);std::string *dest=newdata;std::string *src=elements;for(size_t i=0;i!=size();++i)alloc.construct(dest++,std::move(*src++));free();elements=newdata;first_free=dest;cap=elements+newcapacity;
}
void StrVec::reserve(const size_t n)
{if(n>capacity()){std::string *newdata=alloc.allocate(n);std::string *dest=newdata;std::string *src=elements;for(size_t i=0;i!=size();++i)alloc.construct(dest++,std::move(*src++));free();elements=newdata;first_free=dest;cap=elements+n;}
}
void StrVec::resize(const size_t n,const std::string &orgv)
{if(n<=size()){std::string *b=elements+n;std::string *e=first_free;while(b!=e)alloc.destroy(b++);first_free=elements+n;}else if(n<=capacity()){std::string *b=first_free;std::string *e=elements+n;while(b!=e)alloc.construct(b++,orgv);first_free=e;}else{reserve(n);std::string *b=first_free;std::string *e=elements+n;while(b!=e)alloc.construct(b++,orgv);first_free=e;}
}

13.43

 我更倾向于指针的表达方式,目的更清晰。

void StrVec::free()
{if(elements){for_each(elements,first_free,[](string &str) {alloc.destroy(&str);});alloc.deallocate(elements,cap-elements);}
}

13.44

String.h

#ifndef _sTRING_H
#define _sTRING_H
#include <iostream>
#include <utility>
class String
{friend std::istream & operator>>(std::istream &in,String &str);friend std::ostream & operator<<(std::ostream &out,const String &str);public:String():head(nullptr),tail(nullptr),cap(nullptr) {}String(const char *);String(const String &);String & operator=(const String &);String operator+(const String &);~String() {free();}size_t size() const {return tail-head;}size_t capacity() const {return cap-head;}bool empty() const {return (head==tail)?true:false;}char * begin() const {return head;}char * end() const {return tail;}private:char *head;char *tail;char *cap;static std::allocator<char> alloc;void free();std::pair<char *,char *> alloc_n_copy(const char *,const char *);void chk_n_alloc(size_t n) {if(n+size()>capacity()) reallocate(n);}void reallocate(size_t n);
};
std::istream & operator>>(std::istream &in,String &str);
std::ostream & operator<<(std::ostream &out,const String &str);
#endif==String.cpp==
#include <memory>
#include <algorithm>
#include <cstring>
#include <string>
#include "String.h"String::String(const char *str)
{size_t n=strlen(str);std::pair<char *,char *> newdata=alloc_n_copy(str,str+n);head=newdata.first;tail=cap=newdata.second;
}
String::String(const String &s)
{std::cout<<"This is copy constructor."<<std::endl;std::pair<char *,char *> newdata=alloc_n_copy(s.begin(),s.end());head=newdata.first;tail=cap=newdata.second;
}
String & String::operator=(const String &rhs)
{std::cout<<"This is copy assignment operator."<<std::endl;std::pair<char *,char *> newdata=alloc_n_copy(rhs.begin(),rhs.end());free();head=newdata.first;tail=cap=newdata.second;return *this;
}
String String::operator+(const String &rhs)
{String tmp(*this);size_t n=rhs.size();tmp.chk_n_alloc(n);char *dest=tmp.tail;char *src=rhs.begin();for(size_t i=0;i!=6;++i)alloc.construct(dest++,*src++);tmp.tail=dest;return tmp;
}std::allocator<char> String::alloc;
void String::free()
{if(head){std::for_each(begin(),end(),[](char &c) {alloc.destroy(&c);});alloc.deallocate(head,cap-head);}
}
std::pair<char *,char *> String::alloc_n_copy(const char *b,const char *e)
{char *newdata=alloc.allocate(e-b);return {newdata,std::uninitialized_copy(b,e,newdata)};
}
void String::reallocate(size_t n)
{size_t newcap=(size()>n?size()*2:n*2);char *newdata=alloc.allocate(newcap);char *dest=newdata;char *src=head;for(size_t i=0;i!=size();++i)alloc.construct(dest++,std::move(*src++));free();head=newdata;tail=dest;cap=head+newcap;
}
std::istream & operator>>(std::istream &in,String &str)
{std::string tmp;in>>tmp;String Stmp(tmp.c_str());str=Stmp;return in;
}
std::ostream & operator<<(std::ostream &out,const String &str)
{std::for_each(str.begin(),str.end(),[&out](const char &c)->void {out<<c;});return out;
}

13.45

 右值引用就是必须绑定到右值的引用,通过&&符号获得。右值引用只能绑定到一个即将要销毁的对象上,因此可以自由的移动其源资源。左值引用不能绑定到要转换的表达式、字面常量或返回右值的表达式。而右值引用恰好相反。
返回左值的表达式包括,返回左值引用的函数、赋值、下标、解引用和前置递增/递减运算符。返回右值的表达式包括返回非引用类型的函数、算术、关系、位运算以及后置递增/递减运算符。

13.46

  • int &&r1=f();
  • int &r2=vi[0];
  • int &r3=r1;
  • int &&r4=vi[0]*f();

13.47

见13.44

13.48

#include <iostream>
#include <vector>
#include "13-44/String.h"
using namespace std;
int main(void)
{vector<String> vs;String a("a1"),b("b2"),c("c3");vs.push_back(a);vs.push_back(b);vs.push_back(c);return 0;
}

13.49

StrVec.h

#ifndef _STRVEC_H
#define _STRVEC_H
#include <string>
#include <cstddef>
#include <utility>
#include <memory>
#include <initializer_list>
class StrVec
{public:StrVec():elements(nullptr),first_free(nullptr),cap(nullptr) {}StrVec(const StrVec &);StrVec(StrVec &&s) noexcept :elements(s.elements),first_free(s.first_free),cap(s.cap) {s.elements=s.first_free=s.cap=nullptr;}StrVec(std::initializer_list<std::string>);StrVec & operator=(const StrVec &);StrVec & operator=(StrVec &&) noexcept;~StrVec() {free();}void push_back(const std::string &);size_t size() {return first_free-elements;}size_t capacity() {return cap-elements;}std::string *begin() const {return elements;}std::string *end() const {return first_free;}void resize(const size_t n,const std::string &orgv=std::string());void reserve(const size_t n);private:std::string *elements;std::string *first_free;std::string *cap;static std::allocator<std::string> alloc;void chk_n_alloc() {if(size()==capacity()) reallocate();}std::pair<std::string *,std::string *> alloc_n_copy(const std::string *,const std::string *);void free();void reallocate();
};
#endif

StrVec.cpp

#include "StrVec.h"
std::allocator<std::string> StrVec::alloc;
StrVec::StrVec(const StrVec &org)
{std::pair<std::string *,std::string *> data=alloc_n_copy(org.begin(),org.end());elements=data.first;first_free=cap=data.second;
}
StrVec::StrVec(std::initializer_list<std::string> ls)
{auto data=alloc_n_copy(ls.begin(),ls.end());elements=data.first;first_free=cap=data.second;
}
StrVec & StrVec::operator=(const StrVec &rhs)
{auto data=alloc_n_copy(rhs.begin(),rhs.end());free();elements=data.first;first_free=cap=data.second;return *this;
}
StrVec & StrVec::operator=(StrVec &&s) noexcept
{if(this!=&s){free();elements=s.elements;first_free=s.first_free;cap=s.cap;s.elements=s.first_free=s.cap=nullptr;}return *this;
}
void StrVec::push_back(const std::string &s)
{chk_n_alloc();alloc.construct(first_free++,s);
}
std::pair<std::string *,std::string *> StrVec::alloc_n_copy(const std::string *b,const std::string *e)
{std::string *data=alloc.allocate(e-b);return {data,uninitialized_copy(b,e,data)};
}
void StrVec::free()
{if(elements){for(auto p=first_free;p!=elements;)alloc.destroy(--p);alloc.deallocate(elements,cap-elements);}
}
void StrVec::reallocate()
{size_t newcapacity=size()?2*size():1;std::string *newdata=alloc.allocate(newcapacity);std::string *dest=newdata;std::string *src=elements;for(size_t i=0;i!=size();++i)alloc.construct(dest++,std::move(*src++));free();elements=newdata;first_free=dest;cap=elements+newcapacity;
}
void StrVec::reserve(const size_t n)
{if(n>capacity()){std::string *newdata=alloc.allocate(n);std::string *dest=newdata;std::string *src=elements;for(size_t i=0;i!=size();++i)alloc.construct(dest++,std::move(*src++));free();elements=newdata;first_free=dest;cap=elements+n;}
}
void StrVec::resize(const size_t n,const std::string &orgv)
{if(n<=size()){std::string *b=elements+n;std::string *e=first_free;while(b!=e)alloc.destroy(b++);first_free=elements+n;}else if(n<=capacity()){std::string *b=first_free;std::string *e=elements+n;while(b!=e)alloc.construct(b++,orgv);first_free=e;}else{reserve(n);std::string *b=first_free;std::string *e=elements+n;while(b!=e)alloc.construct(b++,orgv);first_free=e;}
}

String.h

#ifndef _sTRING_H
#define _sTRING_H
#include <iostream>
#include <utility>
class String
{friend std::istream & operator>>(std::istream &in,String &str);friend std::ostream & operator<<(std::ostream &out,const String &str);public:String():head(nullptr),tail(nullptr),cap(nullptr) {}String(const char *);String(const String &);String(String &&str) noexcept :head(str.head),tail(str.tail),cap(str.cap) {str.head=str.tail=str.cap=nullptr;std::cout<<"This is move constructor."<<std::endl;}String & operator=(const String &);String & operator=(String &&);String operator+(const String &);~String() {free();}size_t size() const {return tail-head;}size_t capacity() const {return cap-head;}bool empty() const {return (head==tail)?true:false;}char * begin() const {return head;}char * end() const {return tail;}private:char *head;char *tail;char *cap;static std::allocator<char> alloc;void free();std::pair<char *,char *> alloc_n_copy(const char *,const char *);void chk_n_alloc(size_t n) {if(n+size()>capacity()) reallocate(n);}void reallocate(size_t n);
};
std::istream & operator>>(std::istream &in,String &str);
std::ostream & operator<<(std::ostream &out,const String &str);
#endif

String.cpp

#include <memory>
#include <algorithm>
#include <cstring>
#include <string>
#include "String.h"String::String(const char *str)
{size_t n=strlen(str);std::pair<char *,char *> newdata=alloc_n_copy(str,str+n);head=newdata.first;tail=cap=newdata.second;
}
String::String(const String &s)
{std::cout<<"This is copy constructor."<<std::endl;std::pair<char *,char *> newdata=alloc_n_copy(s.begin(),s.end());head=newdata.first;tail=cap=newdata.second;
}
String & String::operator=(const String &rhs)
{std::cout<<"This is copy assignment operator."<<std::endl;std::pair<char *,char *> newdata=alloc_n_copy(rhs.begin(),rhs.end());free();head=newdata.first;tail=cap=newdata.second;return *this;
}
String & String::operator=(String &&rhs)
{std::cout<<"This is move assignment operator."<<std::endl;if(this!=&rhs){free();head=rhs.head;tail=rhs.tail;cap=rhs.cap;rhs.head=rhs.tail=rhs.cap=nullptr;}return *this;
}
String String::operator+(const String &rhs)
{String tmp(*this);size_t n=rhs.size();tmp.chk_n_alloc(n);char *dest=tmp.tail;char *src=rhs.begin();for(size_t i=0;i!=6;++i)alloc.construct(dest++,*src++);tmp.tail=dest;return tmp;
}std::allocator<char> String::alloc;
void String::free()
{if(head){std::for_each(begin(),end(),[](char &c) {alloc.destroy(&c);});alloc.deallocate(head,cap-head);}
}
std::pair<char *,char *> String::alloc_n_copy(const char *b,const char *e)
{char *newdata=alloc.allocate(e-b);return {newdata,std::uninitialized_copy(b,e,newdata)};
}
void String::reallocate(size_t n)
{size_t newcap=(size()>n?size()*2:n*2);char *newdata=alloc.allocate(newcap);char *dest=newdata;char *src=head;for(size_t i=0;i!=size();++i)alloc.construct(dest++,std::move(*src++));free();head=newdata;tail=dest;cap=head+newcap;
}
std::istream & operator>>(std::istream &in,String &str)
{std::string tmp;in>>tmp;String Stmp(tmp.c_str());str=Stmp;return in;
}
std::ostream & operator<<(std::ostream &out,const String &str)
{std::for_each(str.begin(),str.end(),[&out](const char &c)->void {out<<c;});return out;
}

Message.h

#ifndef _MESSAGE_H
#define _MESSAGE_H
#include <string>
#include <set>
class Folder;
class Message
{friend class Folder;friend void swap(Message &,Message &);public:explicit Message(const std::string &str=""):contents(str) {}Message(const Message &);Message(Message &&m):contents(std::move(m.contents)) {move_Folders(&m);}Message & operator=(const Message &);Message & operator=(Message &&);~Message();void save(Folder &);void remove(Folder &);private:std::string contents;std::set<Folder *> folders;void add_to_Folders(const Message &);void remove_from_Folders();void addfolder(Folder *f) {folders.insert(f);}void remfolder(Folder *f) {folders.erase(f);}void move_Folders(Message *m);
};
class Folder
{friend void swap(Message &,Message &);friend class Message;public:Folder()=default;Folder(const Folder &f):messages(f.messages) {add_to_Messages(f);}Folder(Folder &&f) {move_Messages(&f);}Folder & operator=(const Folder &);Folder & operator=(Folder &&);~Folder();void save(Message &);void remove(Message &);private:std::set<Message *> messages;void addMsg(Message *m) {messages.insert(m);}void remMsg(Message *m) {messages.erase(m);}void add_to_Messages(const Folder &);void remove_from_Messages();void move_Messages(Folder *);
};
#endif

Message.cpp

#include "Message.h"
void Message::save(Folder &f)
{folders.insert(&f);f.addMsg(this);
}
void Message::remove(Folder &f)
{folders.erase(&f);f.remMsg(this);
}
void Message::add_to_Folders(const Message &m)
{for(auto f:m.folders)f->addMsg(this);
}
void Message::remove_from_Folders()
{for(auto f:folders)f->remMsg(this);
}
void Message::move_Folders(Message *m)
{folders=std::move(m->folders);for(Folder *f:folders){f->remMsg(m);f->addMsg(this);}m->folders.clear();
}
Message::Message(const Message &m):contents(m.contents),folders(m.folders)
{add_to_Folders(m);
}
Message & Message::operator=(const Message &rhs)
{remove_from_Folders();contents=rhs.contents;folders=rhs.folders;add_to_Folders(rhs);return *this;
}
Message & Message::operator=(Message &&rhs)
{if(this!=&rhs){remove_from_Folders();contents=std::move(rhs.contents);move_Folders(&rhs);}return *this;
}
Message::~Message()
{remove_from_Folders();
}
void swap(Message &lhs,Message &rhs)
{using std::swap;for(auto f:lhs.folders)f->remMsg(&lhs);for(auto f:rhs.folders)f->remMsg(&rhs);swap(lhs.contents,rhs.contents);swap(lhs.folders,rhs.folders);for(auto f:lhs.folders)f->addMsg(&lhs);for(auto f:rhs.folders)f->addMsg(&rhs);
}void Folder::add_to_Messages(const Folder &f)
{for(auto m:f.messages)m->addfolder(this);
}
void Folder::remove_from_Messages()
{for(auto m:messages)m->folders.erase(this);
}
void Folder::move_Messages(Folder *f)
{messages=std::move(f->messages);for(Message *m:messages){m->remfolder(f);m->addfolder(this);}f->messages.clear();
}
void Folder::save(Message &m)
{messages.insert(&m);m.addfolder(this);
}
void Folder::remove(Message &m)
{messages.erase(&m);m.remfolder(this);
}
Folder & Folder::operator=(const Folder &rhs)
{remove_from_Messages();messages=rhs.messages;add_to_Messages(rhs);return *this;
}
Folder & Folder::operator=(Folder &&rhs)
{if(this!=&rhs){remove_from_Messages();move_Messages(&rhs);}return *this;
}
Folder::~Folder()
{remove_from_Messages();
}

13.50

String代码见此
测试代码

#include <iostream>
#include <vector>
#include "13-44/String.h"
using namespace std;
int main(void)
{vector<String> vs;vs.reserve(6);String a("a1"),b("b2"),c("c3");vs.push_back(a);vs.push_back(std::move(b));vs.push_back(std::move(c));vs.push_back(String("hi"));vs.push_back("hello");return 0;
}

13.51

 它以值的方式返回一个unique_ptr或者返回一个局部对象的拷贝,返回的是右值,实际上执行的是它的移动构造函数。

13.52

hp=std::move(hp2),首先std::move调用获得hp2的右值引用,HasPtr类的赋值运算符是传值类型的形参,所以调用移动构造函数构造rhs,完成后rhs内ps及i均为原hp2内的值,而hp2内ps指向NULL。此后对this及rhs调用swap函数交换两个对象的值,返回this,而rhs离开作用域,调用析构函数。

13.53

#include <iostream>
#include <string>
#include <vector>
#include <algorithm>
class HasPtr
{friend void swap(HasPtr &,HasPtr &);friend std::ostream & print(std::ostream &,const HasPtr &);public:HasPtr(const std::string &s=std::string()):pts(new std::string(s)) {}HasPtr(const HasPtr &org):value(org.value),pts(new std::string(*org.pts)) {}HasPtr(HasPtr &&org) noexcept :value(org.value),pts(org.pts) {org.pts=nullptr;}HasPtr & operator=(const HasPtr &rhs) {std::string *tmp=new std::string(*rhs.pts);delete pts;pts=tmp;value=rhs.value;return *this;}HasPtr & operator=(HasPtr &&rhs) noexcept;bool operator<(const HasPtr &rhs) const {return *pts<*rhs.pts;}~HasPtr() {delete pts;}private:int value=0;std::string *pts;
};
inline HasPtr & HasPtr::operator=(HasPtr &&rhs) noexcept
{if(this!=&rhs){delete pts;pts=rhs.pts;value=rhs.value;rhs.pts=nullptr;}return *this;
}
inline void swap(HasPtr &lhs,HasPtr &rhs)
{using std::swap;swap(lhs.value,rhs.value);swap(lhs.pts,rhs.pts);
}
std::ostream & print(std::ostream &os,const HasPtr &hp)
{os<<*hp.pts<<std::endl;return os;
}

 拷贝并交换版本中效率不理想的原因在于rhs是传值,不管是拷贝构造还是移动构造都要多进行一次。

13.54

#include <iostream>
#include <string>
#include <vector>
#include <algorithm>
class HasPtr
{friend void swap(HasPtr &,HasPtr &);friend std::ostream & print(std::ostream &,const HasPtr &);public:HasPtr(const std::string &s=std::string()):pts(new std::string(s)) {}HasPtr(const HasPtr &org):value(org.value),pts(new std::string(*org.pts)) {}HasPtr(HasPtr &&org) noexcept :value(org.value),pts(org.pts) {org.pts=nullptr;}HasPtr & operator=(HasPtr rhs) {swap(*this,rhs);return *this;}HasPtr & operator=(HasPtr &&rhs) noexcept;bool operator<(const HasPtr &rhs) const {return *pts<*rhs.pts;}~HasPtr() {delete pts;}private:int value=0;std::string *pts;
};
inline HasPtr & HasPtr::operator=(HasPtr &&rhs) noexcept
{if(this!=&rhs){delete pts;pts=rhs.pts;value=rhs.value;rhs.pts=nullptr;}return *this;
}
inline void swap(HasPtr &lhs,HasPtr &rhs)
{using std::swap;swap(lhs.value,rhs.value);swap(lhs.pts,rhs.pts);
}
std::ostream & print(std::ostream &os,const HasPtr &hp)
{os<<*hp.pts<<std::endl;return os;
}
int main(void)
{HasPtr h("hi mom!");HasPtr h2(h);HasPtr h3 = h;h2 = h3;h2 = std::move(h3);return 0;
}

 无法通过编译,运算符重载存在二义性。

13.55

StrBlob.h

#ifndef STRBLOB_H
#define STRBLOB_H
#include <memory>
#include <vector>
#include <string>
#include <initializer_list>
#include <stdexcept>
class StrBlobPtr;
class StrBlob
{
friend class StrBlobPtr;
public:typedef std::vector<std::string>::size_type size_type;StrBlob();StrBlob(std::initializer_list<std::string> il);StrBlob(const StrBlob &);StrBlob & operator=(const StrBlob &);size_type size() const {return data->size();}bool empty() const {return data->empty();}void push_back(const std::string &t){data->push_back(t);}void push_back(std::string &&t){data->push_back(std::move(t));}void pop_back();std::string &front();std::string &back();const std::string &front() const;const std::string &back() const;StrBlobPtr begin();StrBlobPtr begin() const;StrBlobPtr end();StrBlobPtr end() const;
private:std::shared_ptr<std::vector<std::string>> data;void check(size_type i,const std::string &msg) const;
};
class StrBlobPtr
{
public:StrBlobPtr():curr(0){}StrBlobPtr(StrBlob &a,std::size_t sz=0):wptr(a.data),curr(sz){}StrBlobPtr(const StrBlob &a,std::size_t sz=0):wptr(a.data),curr(sz){}std::string &deref() const;StrBlobPtr &incr();
private:std::shared_ptr<std::vector<std::string>> check(std::size_t,const std::string &) const;std::weak_ptr<std::vector<std::string>> wptr;std::size_t curr=0;
};
#endif

StrBlob.cpp

#include "StrBlob.h"
using namespace std;
StrBlobPtr StrBlob::begin()
{return StrBlobPtr(*this);
}
StrBlobPtr StrBlob::begin() const
{return StrBlobPtr(*this);
}
StrBlobPtr StrBlob::end()
{return StrBlobPtr(*this,data->size());
}
StrBlobPtr StrBlob::end() const
{return StrBlobPtr(*this,data->size());
}StrBlob::StrBlob():data(make_shared<vector<string>>()){}
StrBlob::StrBlob(initializer_list<string> il):data(make_shared<vector<string>>(il)){}
StrBlob::StrBlob(const StrBlob &org):data(make_shared<std::vector<std::string>>(*org.data)) {}
StrBlob & StrBlob::operator=(const StrBlob &rhs)
{data=make_shared<std::vector<std::string>>(*rhs.data);return *this;
}
inline void StrBlob::check(size_type i,const string &msg) const
{if(i>=data->size())throw out_of_range(msg);
}
string &StrBlob::front()
{check(0,"front on empty StrBlob");return data->front();
}
string &StrBlob::back()
{check(0,"back on empty StrBlob");return data->back();
}
const std::string &StrBlob::front() const
{check(0,"front on empty StrBlob");return data->front();
}
const std::string &StrBlob::back() const
{check(0,"back on empty StrBlob");return data->back();
}
void StrBlob::pop_back()
{check(0,"pop_back on empty StrBlob");data->pop_back();
}std::shared_ptr<std::vector<std::string>> StrBlobPtr::check(std::size_t i,const std::string &msg) const
{auto ret=wptr.lock();if(!ret)throw std::runtime_error("unbound StrBlobPtr");if(i>=ret->size())throw std::out_of_range(msg);return ret;
}
std::string & StrBlobPtr::deref() const
{auto p=check(curr,"dereference past end");return (*p)[curr];
}
StrBlobPtr & StrBlobPtr::incr()
{check(curr,"increment past end of StrBlobPtr");++curr;return *this;
}

13.56

ret是左值,依然会调用sorted的左值版本,递归调用无法结束,最后导致栈溢出。

13.57

Foo(*this)是右值,会调用sorted的右值版本,排序正常完成。

13.58

#include <iostream>
#include <vector>
#include <algorithm>
using std::vector;
class Foo
{public:Foo sorted() &&;Foo sorted() const &;private:std::vector<int> data;
};
Foo Foo::sorted() &&
{std::cout<<"右值引用版本"<<std::endl;sort(data.begin(),data.end());return *this;
}
#ifdef _A
Foo Foo::sorted() const &
{std::cout<<"左值引用版本"<<std::endl;Foo ret(*this);return ret.sorted();
}
#else
Foo Foo::sorted() const &
{std::cout<<"左值引用版本"<<std::endl;return Foo(*this).sorted();
}
#endif
int main(void)
{Foo a;a.sorted();return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/730290.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

PostgreSQL教程(二十二):服务器管理(四)之服务器配置

一、设置参数 1.1 参数名称和值 所有参数名都是大小写不敏感的。每个参数都可以接受五种类型之一的值&#xff1a; 布尔、字符串、整数、 浮点数或枚举。该类型决定了设置该参数的语法&#xff1a; 布尔: 值可以被写成 on, off, true, false, yes, no, 1, 0 &#xff08;都是…

Programming Abstractions in C阅读笔记:p312-p326

《Programming Abstractions in C》学习第77天&#xff0c;p312-p326&#xff0c;总计15页&#xff0c;第7章完结。 一、技术总结 第7章主要讲算法分析——引入时间复杂度这一概念来评估算法的快慢。时间复杂度使用大O符号来表示。 第7章以排序算法为示例&#xff0c;包含&a…

go调用 c++中数组指针相关

要在Go语言中调用C编译的DLL&#xff08;动态链接库&#xff09;并传递数组&#xff0c;你需要遵循以下步骤&#xff1a; 编写C代码&#xff1a;首先&#xff0c;你需要有一个C的DLL&#xff0c;它提供了你想要在Go中调用的函数。为了确保Go可以调用它&#xff0c;你需要使用C…

[PTA] 分解质因子

输入一个正整数n&#xff08;1≤n≤1e15&#xff09;&#xff0c;编程将其分解成若干个质因子&#xff08;素数因子&#xff09;积的形式。 输入格式: 任意给定一个正整数n&#xff08;1≤n≤1e15&#xff09;。 输出格式: 将输入的正整数分解成若干个质因子积的形式&#…

ubuntu 卸载miniconda3

一开始安装路径错了&#xff0c;需要重新安一次&#xff0c;就一起记录了。 前提是这种方式安装&#xff1a; ubuntu安装miniconda3管理python版本-CSDN博客 删除Miniconda的安装目录 这目录就是你选择安装的时候指定的&#xff0c;如果记不得了,可以这样查看 which conda 这…

数据库压力测试方法概述

一、前言 在前面的压力测试过程中&#xff0c;主要关注的是对接口以及服务器硬件性能进行压力测试&#xff0c;评估请求接口和硬件性能对服务的影响。但是对于多数Web应用来说&#xff0c;整个系统的瓶颈在于数据库。 原因很简单&#xff1a;Web应用中的其他因素&#xff0c;…

Chrome安装Axure插件

打开原型目录/resources/chrome&#xff0c;重命名axure-chrome-extension.crx&#xff0c;修改后缀为rar&#xff0c;axure-chrome-extension.rar 解压到axure-chrome-extension目录打开Chrome&#xff0c;更多工具->扩展程序&#xff0c;打开开发者模式&#xff0c;选择加…

结构体和malloc学习笔记

结构体学习&#xff1a; 为什么会出现结构体&#xff1a; 为了表示一些复杂的数据&#xff0c;而普通的基本类型变量无法满足要求&#xff1b; 定义&#xff1a; 结构体是用户根据实际需要自己定义的符合数类型&#xff1b; 如何使用结构体&#xff1a; //定义结构体 struc…

[C++]类和对象,explicit,static,友元,构造函数——喵喵要吃C嘎嘎4

希望你开心&#xff0c;希望你健康&#xff0c;希望你幸福&#xff0c;希望你点赞&#xff01; 最后的最后&#xff0c;关注喵&#xff0c;关注喵&#xff0c;关注喵&#xff0c;大大会看到更多有趣的博客哦&#xff01;&#xff01;&#xff01; 喵喵喵&#xff0c;你对我真的…

FineReport决策报表Excel导出数据不全解决办法

一、首先建立决策报表 决策报表不带参数导出办法&#xff08;即没有参数面板&#xff09; 普通决策报表导出&#xff08;没有搜索面板&#xff09; 如果决策报表带参数&#xff08;即有搜索框&#xff09;&#xff0c;用上面的办法只能导出部分数据&#xff0c;数据不全 二、…

如何应对不想工作的情况

不想工作,这是每个人都可能遇到的情况。但是,作为一名程序员,我们需要保持高效率和创造力,以满足项目的需求和进度。以下是一些建议,帮助你在不想工作的时候保持高效和专注。 确定原因 不想工作可能有很多原因,比如疲劳、缺乏动力、工作内容不感兴趣等等。确定原因可以帮助你找…

蓝色经典免费wordpress模板主题

蓝色经典配色的免费wordpress建站主题&#xff0c;万能的wordpress建站主题。 https://www.wpniu.com/themes/24.html

董宇辉所有商标已转到与辉同行名下!

近日董宇辉此前由东方优选申请的所有商标已转到与辉同行主体名下&#xff0c;普推知产老杨经检索发现&#xff0c;这些商标都是2022年6月由东方优选提交申请&#xff0c;在2023年12月28时提交商标转让&#xff0c;最近转让成功&#xff0c;转让周期是2个半月左右。 转让的商标除…

指针进阶(下)指针实操

sizeof 和 strlen 首先我们来复习一下sizeof 和 strlen 的区别。 sizeof 是操作符&#xff0c;只关注内存中存放的数据的大小&#xff0c;并不会参与sizeof 括号内部的计算。注意它的单位是字节 #include <stdio.h>int main() {int a 10;printf("%d\n", size…

作业1-32 B3620 x 进制转 10 进制

题目 思路 分析题目可知&#xff0c;此题可以用到大写字母&#xff0c;也就是从A开始&#xff0c;分别表示11往后的数字。 那么就用一个for循环&#xff0c;将零到九划分为一个等级&#xff0c;将A到Z划分为一个等级。 for(int i0;i<str.length();i){if(str[i]>0&&…

Zabbix(四)

Zabbix Proxy zabbix作为一个分布式监控系统(分布式监控解决方案)&#xff0c;支持通过代理(proxy)收集zabbix agent的监控数据&#xff0c;然后由zabbix proxy再把数据发送给zabbix server&#xff0c;也就是zabbix proxy 可以代替zabbix server收集监控数据&#xff0c;然后…

【免费资源】Unity真实广阔的沙漠场景等你来解锁!

Unity真实广阔的沙漠场景等你来解锁&#xff01; Unity 每周免费资源上新啦&#xff01;此次更新的是广阔的沙漠场景&#xff0c;其中包含 14 个预制体&#xff0c;每个预制体都包含 LOD、400-2000 顶点和 4K 纹理。现在&#xff0c;只需登录 Asset Store&#xff0c;即可免费领…

怎么将pom在文件放到src下方

今天在IDEA从git拉取项目的时候&#xff0c;发现pom.xml文件在文件夹src的上方&#xff0c;平时看惯了项目的pom.xml文件在文件夹src的下方&#xff0c;应该怎么去设置呢&#xff1f; 点击设置——>点击Folder Always on Top 即可 参考&#xff1a;http://t.csdnimg.cn/s34…

达梦数据库——如何查看数据库大字段中的数据内容

今天get到一个小知识点 分享给大家&#xff0c;如何在数据库查看大字段中的数据内容。 以下为演示步骤&#xff0c;简单易懂&#xff0c;操练起来吧 首先创建一个含有CLOB、TEXT的大字段测试表 create table "SYSDBA"."CS"("COLUMN_1" CLOB,&qu…

配电网数字化转型全面推进:《关于新形势下配电网高质量发展的指导意见》

近日&#xff0c;国家发展改革委、国家能源局印发了《关于新形势下配电网高质量发展的指导意见》&#xff08;以下简称《意见》&#xff09;&#xff0c;到2030年&#xff0c;基本完成配电网柔性化、智能化、数字化转型&#xff0c;实现主配微网多级协同、海量资源聚合互动、多…