第12周-任务3-做个摩托车开一开

时间:2022-08-04 08:03:11

【题记】看到车辆类、自行车类、机动车类、摩托车类很喜欢,决定利用它们练习多重继承。找到的原题是程序填空题,代码中只将各个类描述完后了事。产生了造一辆能开的摩托车的想法后,一边构思,一边写代码,忙乎到深夜,终于在功能方面,在顾及同学们的能力方面,在满足实验目标方面达到了折衷,有了这道题。日后,这辆在黑框框中跑的车,定能成为同学们做真车(或游戏车)的启发。


【题目】在下面一段类的定义中,自行车类的虚基类为车辆类,机动车类的虚基类也为车辆类,摩托车类的基类为自行车类和机动车类,类之间均为公有继承。
(1)根据上面各类间关系的描述,补全下面程序段中空缺的代码
(2)实现程序中声明的成员函数,注意相应操作中的动作发生的条件不能满足时应给出提示。
(3)运行程序,享受开摩托的过程。(BB平台上提供了一个可执行文件,可以先运行再编程。不必申请驾照,这个摩托车很安全。)
(4)在报告中回答问题:本题中使用虚基类的好处是什么?

参考代码:

#include <iostream>
#include<conio.h>
#include <windows.h>
using namespace std;
enum vehicleStaus {rest, running};  //车辆状态:泊车、行进
class vehicle //车辆类
{
protected:
	int maxSpeed;		//最大车速
	int currentSpeed;	//当前速度
	int weight;			//车重
	vehicleStaus status; //rest-泊车状态;running-行进状态
public:
	vehicle(int maxS, int w); //构造函数,初始时,当前速度总为0且处在停车状态
	void start();  //由rest状态到running, 初速为1
	void stop(); //由running状态到rest, 当前速度小于5时,才允许停车
	void speed_up();  //加速,调用1次,速度加1
	void slow_down(); //减速,调用1次,速度减1,速度为0时,停车
};

class bicycle :_____(1)_________//(1)自行车类的虚基类为车辆类
{ 
protected:
	double height; //车高
public:
	bicycle(int maxS=10, int w=50, int h=0.7);   //定义构造函数
};

class motorcar : ______(2)__________//(2)机动车类的虚基类也为车辆类
{ 
protected:
	int seatNum; //座位数
	int passengerNum; //乘客人数
public:
	motorcar(int maxS=150, int w=1500, int s=5, int p=1);   //定义构造函数
	void addPassenger(int p=1);   //增加搭载的乘客,超员要拒载,有人下车时,p为负数。当然车上乘客至少有1个(司机)。只有车停稳后才能上下客。
};

class motorcycle: ______(3)_________ //(3)摩托车类的基类为自行车类和机动车类
{ 
public:
	//定义构造函数
	motorcycle(int maxS=90, int w=100, int s=3, int p=1, int h=0.7);   
	void show(); //显示摩托车的运行状态
};

int main( )
{
	motorcycle m;
	bool end=false;
	while (!end){
		cout<<"请操作:1-启动  2-加速  3-减速  4-有人上车  5-有人下车  6-停车 0-结束"<<endl;
		char keydown= _getch(); //_getch()返回键盘上读取的字符
		switch(keydown)
		{
		case '1': 
			cout<<"操作(启动)\t"; m.start(); break;
		case '2':                         
			cout<<"操作(加速)\t"; m.speed_up(); break;
		case '3':                          
			cout<<"操作(减速)\t"; m.slow_down(); break;
		case '4':                        
			cout<<"操作(有人上车)\t"; m.addPassenger(); break;
		case '5':                      
			cout<<"操作(有人下车)\t"; m.addPassenger(-1); break;
		case '6':                      
			cout<<"操作(停车)\t"; m.stop(); break;
		case '0':               
			end=true; break;
		}
		m.show();
		cout<<endl;
		Sleep(200);  //要包含头文件<windows.h>
	}
	system("pause");
	return 0;
}
运行截图:

第12周-任务3-做个摩托车开一开

【简析】把握好四个类之间的继承关系:

第12周-任务3-做个摩托车开一开

  在这种继承关系中,一个派生类有多个直接基类,而这些直接基类又有一个共同的基类。如果不做特殊处理,在最终的派生类中保留间接共同基类数据成员的多份同名成员,产生二义性问题。利用C++提供虚基类(virtual base class)的方法,使得在继承间接共同基类时只保留一份成员。在此,声明bicycle和motorcar类时,将其基类vehical声明为虚基类,注意语法即可。(突然想起人类社会中的“隔代亲”。跨过了父一代,孙子直接享用爷爷的成员。)

  另外,本题的各个成员函数在修改数据成员值时,要考虑到必要的约束:如上下车要停车、不能超速和超载等。


【参考代码】

#include <iostream>
#include<conio.h>
#include <windows.h>
using namespace std;
enum vehicleStaus {rest, running};  //车辆状态:泊车、行进
class vehicle //车辆类
{
protected:
	int maxSpeed;		//最大车速
	int currentSpeed;	//当前速度
	int weight;			//车重
	vehicleStaus status; //rest-泊车状态;running-行进状态
public:
	vehicle(int maxS, int w); //构造函数,初始时,当前速度总为0且处在停车状态
	void start();  //由rest状态到running, 初速为1
	void stop(); //由running状态到rest, 当前速度小于5时,才允许停车
	void speed_up();  //加速,调用1次,速度加1
	void slow_down(); //减速,调用1次,速度减1,速度为0时,停车
};

//构造函数,初始时,当前速度总为0且处在停车状态
vehicle::vehicle(int maxS, int w):maxSpeed(maxS), currentSpeed(0),weight(w), status(rest){}

//启动:由rest状态到running, 初速为1
void vehicle::start()
{
	if (status==rest)
	{
		status=running;
		currentSpeed=1;
	}
	else
		cout<<"车辆已经行驶!"<<endl;
}
//由running状态到rest, 当前速度小于5时,才允许停车
void vehicle::stop()
{
	if (status==running)
		if(currentSpeed<5)
		{
			status=rest;
			currentSpeed=0;
		}
		else
			cout<<"车速太快!先减速再停车……"<<endl;
	else
		cout<<"车辆未启动!"<<endl;
}

//加速,调用1次,速度加1
void vehicle::speed_up()
{
	if (status==running)
		if(currentSpeed<maxSpeed)
			++currentSpeed;
		else
			cout<<"请不要超速行驶……"<<endl;
	else
		cout<<"车辆未启动!"<<endl;
}
//减速,调用1次,速度减1,速度为0时,停车
void vehicle::slow_down()
{
	if (status==running)
	{
		if(currentSpeed>0)
			--currentSpeed;
	}
	else
		cout<<"车辆未启动!"<<endl;
	if(currentSpeed==0)
		status=rest;
}

class bicycle :virtual public vehicle //()自行车类
{ 
protected:
	double height; //车高
public:
	bicycle(int maxS=10, int w=50, int h=0.7);   //定义构造函数
};

bicycle::bicycle(int maxS, int w, int h):vehicle(maxS, w),height(h){} 

class motorcar : virtual public vehicle//()机动车类
{ 
protected:
	int seatNum; //座位数
	int passengerNum; //乘客人数
public:
	motorcar(int maxS=150, int w=1500, int s=5, int p=1);   //定义构造函数
	void addPassenger(int p=1);   //搭载乘客,超员要拒载,有人下车时,p为负数。当然车上乘客至少有1个(司机)。上下车时要保证安全……
};

//定义构造函数
motorcar::motorcar(int maxS, int w, int s, int p): vehicle(maxS, w),seatNum(s),passengerNum(p){} 

//搭载乘客,超员要拒载,有人下车时,p为负数。当然车上乘客至少有1个(司机)。上下车时要保证安全……
void motorcar::addPassenger(int p)
{
	if (status==running)
	{
		cout<<"车辆正在行驶,停车后再上下车!"<<endl;
	}
	else
	{
		passengerNum+=p;
		if(passengerNum>seatNum)
		{
			passengerNum=seatNum;
			cout<<"涉嫌超员,已清理后达到满员!"<<endl;
		}
		else if (passengerNum<1)
		{
			passengerNum=1;
			cout<<"请司机不要离开岗位!"<<endl;
		}
	}
}

class motorcycle: public bicycle, public motorcar //()摩托车类
{ 
public:
	motorcycle(int maxS=90, int w=100, int s=3, int p=1, int h=0.7);   //定义构造函数
	void show(); //显示摩托车的运行状态
};

//定义构造函数
motorcycle::motorcycle(int maxS, int w, int s, int p, int h):vehicle(maxS, w),bicycle(maxS, w, h),motorcar(maxS, w, s, p){}

//显示摩托车的运行状态
void motorcycle::show()
{
	cout<<"状态:";
	if(status==rest)
		cout<<"泊车;\t";
	else
		cout<<"行进;\t";
	cout<<"车速:"<<currentSpeed<<" / "<< maxSpeed <<"\t当前乘员:"<<passengerNum<<" / "<< seatNum << endl;
}

int main( )
{
	motorcycle m;
	bool end=false;
	while (!end){
		cout<<"请操作:1-启动  2-加速  3-减速  4-有人上车  5-有人下车  6-停车 0-结束"<<endl;
		char keydown= _getch(); //_getch()返回键盘上读取的字符,应包含头文件<conio.h>
		switch(keydown)
		{
		case '1': 
			cout<<"操作(启动)\t"; m.start(); break;
		case '2':                         
			cout<<"操作(加速)\t"; m.speed_up(); break;
		case '3':                          
			cout<<"操作(减速)\t"; m.slow_down(); break;
		case '4':                        
			cout<<"操作(有人上车)\t"; m.addPassenger(); break;
		case '5':                      
			cout<<"操作(有人下车)\t"; m.addPassenger(-1); break;
		case '6':                      
			cout<<"操作(停车)\t"; m.stop(); break;
		case '0':               
			end=true; break;
		}
		m.show();
		cout<<endl;
		Sleep(200);  //要包含头文件<windows.h>
	}
	system("pause");
	return 0;
}