练习7.16 在类的定义中对于访问说明符出现的位置和次数有限定吗?如果有,是什么?什么样的成员应该定义在public说明符之后?什么样的成员应该定义在private说明符之后?
访问说明符出现的次数和位置没有限制。
作为接口的一部分,构造函数和部分成员函数紧跟在public说明符之后;而数据成员和实现部分的函数则跟在private说明符后面。
练习7.17 使用class和struct时有区别吗?如果有,是什么?
有区别,默认的访问权限是不一样的。
练习7.18 封装是何含义?它有什么用处?
定义在private说明符之后的成员可以被类的成员函数访问,但是不能使用该类的代码访问,private部分封装了(即隐藏了)类的实现细节。
练习7.19 在你的Person类中,你将哪些成员声明成public?哪些声明成private?解释你这样做的原因。
class Person
{
public:
Person() = default;
Person(std::string n):name(n){}
Person(std::string n, std::string a):name(n),adress(a){}
std::string isname() const { return name; };
std::string isadress() const { return adress; };
private:
std::string name;
std::string adress;
};
将数据成员放在private中。
7.20 有友元在什么时候有用?请分别列举出使用友元的利弊。
友元允许其他类以及函数访问非公有成员。
利:能够访问私有成员;弊:破坏了封装性。
7.21 修改你的Sales_data类使其隐藏实现的细节。你之前编写的关于Sales_data操作的程序应该继续使用,借助类的新定义重新编译该程序,确保其工作正常。
class Sales_data
{
friend std::istream &read(std::istream&, Sales_data&);
friend std::ostream &print(std::ostream&, const Sales_data&);
friend Sales_data add(const Sales_data&, const Sales_data&);
public:
Sales_data() = default;
Sales_data(std::string s):book_no(s){}
Sales_data(std::string s, unsigned n, double p) :book_no(s), book_sold(n), revenue(n*p) {}
Sales_data(std::istream &);
std::string isbn() const { return book_no; }
Sales_data& combine (const Sales_data&);
private:
double avg_price() const
{
return book_sold? revenue / book_sold : 0;
}
std::string book_no;
unsigned book_sold = 0;
double revenue = 0.0;
};
7.22 修改你的Person类使其隐藏实现的细节。
class Person
{
friend std::ostream &print(std::ostream &os, const Person &m);
friend std::ostream &print(std::ostream &os, const Person &m);
public:
Person() = default;
Person(std::string n):name(n){}
Person(std::string n, std::string a) :name(n), adress(a) {}
std::string isname() const { return name; };
std::string isadress() const { return adress; };
private:
std::string name;
std::string adress;
};