}
};
int main()
{
Personp("张少华", "420106196611070538");
Teachert("李若山", "420106195801247168","教授", 5000);
p.Print();
t.Print();
}
同步练习8.5
一、选择题
1.当不同的类具有相同的间接基类时,( )。
(A)各派生类无法按继承路线产生自己的基类版本
(B)为了建立唯一的间接基类版本,应该声明间接基类为虚基类
(C)为了建立唯一的间接基类版本,应该声明派生类虚继承基类
(D)一旦声明虚继承,基类的性质就改变了,不能再定义新的派生类
2.下列关于多继承的描述,错误的是( )。
(A)一个派生类对象可以拥有多个直接或间接基类的成员
(B)在多继承时不同的基类可以有同名成员
(C)对于不同基类的同名成员,派生类对象访问它们时不会出现二义性
(D)对于不同基类的不同名成员,派生类对象访问它们时不会出现二义性
3.下面关于基类和派生类的描述,正确的是( )。
(A)一个类可以被多次说明为一个派生类的直接基类,可以不止一次地成为间接基类
(B)一个类不能被多次说明为一个派生类的直接基类,可以不止一次地成为间接基类
(C)一个类不能被多次说明为一个派生类的直接基类,且只能成为一次间接基类
(D)一个类可以被多次说明为一个派生类的直接基类,但只能成为一次间接基类
4.下列关于虚继承的说明形式的描述,正确的是( )。
(A)在派生类类名前添加关键字virtual (B)在基类类名前添加关键字virtual
(C)在基类类名后添加关键字virtual
(D)在派生类类名后,类继承的关键字之前添加关键字virtual
5.设置虚基类的目的是( )。
(A)简化程序 (B)消除二义性 (C)提高运行效率 (D)减少目标代码
【解答】 C C B D B
二、程序练习
1.阅读程序,写出运行结果。
#include
using namespace std;
class A
{ public :
A(constchar *s) { cout << s << endl; }
~A() {}
};
class B : virtual public A
{ public :
B(constchar *s1, const char *s2) : A( s1 ) { cout << s2 << endl; }
};
class C : virtual public A
{ public :
C(const char *s1, const char *s2):A(s1) { cout<< s2 << endl; }
};
class D : public B, public C
{ public :
D( constchar *s1,const char *s2,const char *s3,const char *s4 ):
B(s1, s2 ), C( s1, s3 ), A( s1 )
{ cout << s4 << endl; }
};
int main()
{ D *ptr = newD( "class A", "class B", "class C", "classD" );
delete ptr;
}
【解答】
综合练习
一、思考题
1.函数和类这两种程序模块都可以实现软件重用,它们之间有什么区别?
【解答】
函数是基于参数集的功能抽象模块,以调用方式实现软件重用,函数之间没有逻辑关系。
类是数据属性与操作的封装,以继承方式实现软件重用,类之间构成有向无回图的类格。
2.按照类成员的访问特性、类层次的继承特点,制作一张表格,总结各种类成员在基类、派生类中的可见性和作用域。
【解答】
基类成员 派生类继承 |
public |
protected |
private |
public |
在派生类中访问特性不变。派生类和类外均可见,有作用域。 |
在派生类中访问特性不变。类体系中可见。 |
基类私有成员,仅在基类中可见。 |
protected |
成为派生类保护段成员。在整个类体系中可见。 |
private |
成为派生类私有成员。仅在派生类和基类中可见。 |
|
派生类不论以何种方式继承基类,基类所有成员在整个类体系有作用域。 |
3.若有以下说明语句:
class A
{ private : inta1;
public : int a2; double x;
/*…*/
};
class B : private A
{ private : intb1;
public : int b2; double x;
/*…*/
};
B b;
对象b将会生成什么数据成员?与继承关系、访问特性、名字有关吗?
【解答】
对象b生成的数据成员有a1 a2 A::x b1 b2 B::x,共六个数据成员。数据成员的建立和继承关系、访问特性、名字无关。
4.若有以下说明语句:
class A
{ /*…*/
public : void sameFun();
};
class B : public A
{ /*…*/
public : void sameFun();
};
void comFun()
{ A a;
B b;
/*…*/
}
(1)若在B::sameFun中调用A::sameFun,语句格式如何?它将在什么对象上操作?
(2)在comFun中可以用什么方式调用A::sameFun和B::sameFun?语句格式如何?它们将可以在什么对象上操作?
【解答】
(1)若要在B::sameFun中调用A::sameFun,语句形式应为:
A::samefun(); //域作用符说明调用基类函数
调用的A::samefun将在B类对象上进行操作。
(2)在comFun中调用B::sameFun和A::sameFun的方式有:
a.A::sameFun(); //通过A类对象调用A::sameFun,在a类对象上操作
b.sameFun(); //通过B类对象调用B::sameFun,在b类对象上操作
b.A::sameFun(); //通过B类对象调用A::sameFun
//在b类对象上(对由基类继承下来的数据成员)操作
5.有人定义一个教师类派生一个学生类。他认为“姓名”和“性别”是教师、学生共有的属性,声明为public,“职称”和“工资”是教师特有的,声明为private。在学生类中定义特有的属性“班级”和“成绩”。所以有:
class teacher
{ public:
char name[20]; char sex;
//…
private:
char title[20]; doublesalary;
};
class student : public teacher
{ //…
private:
char grade[20]; int score;
};
你认为这样定义合适吗?请给出你认为合理的类结构定义。
【解答】
不合适,这样导致数据冗余。合理的结构是提取它们共有的数据和操作定义一个基类,然后分别定义teacher和student作为派生类。
class person
{ protected:
charname[20]; char sex;
//……
};
class teacher :public teache
{ //……
private:
char title[20]; double salary;
};
class student :public teacher
{ //……
private :
char grade[20] ;int score;
};
6.在第6章的例6-21中,定义Student类包含了Date类成员。可以用继承方式把Student类定义为Date类的派生类吗?如何改写程序?请你试一试。
【解答】
可以用继承方式改写。程序略。
7.“虚基类”是通过什么方式定义的?如果类A有派生类B、C,类A是类B虚基类,那么它也一定是类C的虚基类吗?为什么?
【解答】
虚基类是在声明派生类时,指定继承方式时声明的,声明虚基类的一般形式为:
class 派生类名 : virtual 继承方式 基类名
若类A是类B和类C的虚基类,但不一定是类D的虚基类,原因在于“虚基类”中的“虚”不是基类本身的性质。而是派生类在继承过程中的特性。关键字virtual只是说明该派生类把基类当作虚基类继承,不能说明基类其他派生类继承基类的方式
8.在具有虚继承的类体系中,建立派生类对象时,以什么顺序调用构造函数?请用简单程序验证你的分析。
【解答】
在具有虚继承的类体系中,建立派生类对象时先调用间接基类构造函数,再按照派生类定义时各个直接基类继承的顺序调用直接基类的构造函数,最后再对派生类对象自身构造函数。
另外,C++为了保证虚基类构造函数只被建立对象的类执行一次,规定在创建对象的派生类构造函数中只调用虚基类的构造函数和进行(执行)自身的初始化。参数表中的其他调用被忽略,即直接基类的构造函数只调用系统自带的版本,或调用自定义版本但不对虚基类数据成员初始化。
程序略。
二、程序设计
1.假设某销售公司有一般员工、销售员工和销售经理。月工资的计算办法是:
一般员工月薪=基本工资;
销售员工月薪=基本工资+销售额*提成率;
销售经理月薪=基本工资+职务工资+销售额*提成率。
编写程序,定义一个表示一般员工的基类Employee,它包含三个表示员工基本信息的数据成员:编号number、姓名name和基本工资basicSalary。
由Employee类派生销售员工Salesman类,Salesman类包含两个新数据成员:销售额sales和静态数据成员提成比例commrate。
再由Salesman类派生表示销售经理的Salesmanager类。Salesmanager类包含新数据成员:岗位工资jobSalary。
为这些类定义初始化数据的构造函数,以及输入数据input、计算工资pay和输出工资条print的成员函数。
设公司员工的基本工资是2000元,销售经理的岗位工资是3000元,提成率=5/1000。在main函数中,输入若干个不同类型的员工信息测试你的类结构。
【解答】
#include
using namespacestd;
class Employee
{
public:
Employee( char Snumber[]="\0", char Sname[]="\0",double bSalary=2000 )
{
strcpy_s(number,Snumber);
strcpy_s(name,Sname);
basicSalary=bSalary;
}
void input()
{
cout << "编号:" ; cin>> number;
cout <<"姓名:" ; cin>> name;
}
void print()
{
cout<<"员工 :"<
}
protected:
char number[5];
char name[10];
double basicSalary;
};
class Salesman:public Employee
{
public:
Salesman(int sal=0)
{ sales=sal; }
void input()
{
Employee::input();
cout<<"本月个人销售额:";
cin>>sales;
}
void pay()
{
salary =basicSalary+sales*commrate;
}
void print()
{
pay();
cout<<"销售员 :"<
protected:
static double commrate;
int sales;
double salary;
};
double Salesman:: commrate=0.005;
classSalesmanager : public Salesman
{
public:
Salesmanager(double jSalary=3000)
{
jobSalary = jSalary;
}
void input()
{
Employee::input();
cout<<"本月部门销售额:";
cin>>sales;
}
voidpay()
{
salary = jobSalary +sales*commrate;
}
void print()
{
pay();
cout<<"销售经理 :"<
private:
double jobSalary;
};
int main()
{
cout<<"基本员工\n";
Employee emp1;
emp1.input();
emp1.print();
cout<<"销售员\n";
Salesman emp2;
emp2.input();
emp2.print();
cout<<"销售经理\n";
Salesmanager emp3;
emp3.input();
emp3.print();
}
2.试写出你所能想到的所有形状(包括二维的和三维的),生成一个形状层次类体系。生成的类体系以Shape作为基类,并由此派生出TwoDimShape类和ThreeDimShape类。它们的派生类是不同的形状类。定义类体系中的每个类,并用main函数进行测试。
【解答】
略。
3.为第7章综合练习的程序设计第1题和第2题中的Integer类和Real类定义一个派生类IntReal:
class IntReal : public Integer, public Real;
使其可以进行+、-、*、/、= 的左、右操作数类型不同的相容运算,并符合原有运算类型转换的语义规则。
【解答】
略。
4.使用Integer类,定义派生类Vector类:
class Integer
{ //…
protected :
int n;
};
class Vector:public Integer
{ //…
protected :
int *v;
};
其中,数据成员v用于建立向量,n为向量长度。要求:类的成员函数可以实现向量的基本算术运算。
【解答】
略。
5.用包含方式改写第4题中的Vector类,使其实现相同的功能。
class Vector
{ //…
protected :
Integer *v;
Integersize;
};
【解答】
略。
6.使用第5题定义的Vector类,定义它的派生类Matrix,实现矩阵的基本算术运算。
【解答】
略。
7.用包含方式改写第6题的Matrix类,使其实现相同的功能。
【解答】
略。
8.设计快捷店会员的简单管理程序。基本要求如下:
(1)定义人民币RMB类,实现人民币的基本运算和显示。
(2)定义会员member类,表示会员的基本信息,包括:编号(按建立会员的顺序自动生成),姓名,密码,电话。提供输入、输出信息等功能。
(3)由RMB类和member类共同派生一个会员卡memberCar类,提供新建会员、充值、消费和查询余额等功能。
(4)main函数定义一个memberCar类数组或链表,保存会员卡,模拟一个快捷店的会员卡管理功能,主要包括:
① 新建会员;
② 已有会员充值;
③ 已有会员消费(凭密码,不能透支);
④ 输出快捷店当前会员数,营业额(收、支情况)。
【解答】
略。