实验十 多态:抽象类
1 实验目的
学习为什么要使用抽象类;
学习通过继承,实现代码重用的机制和方法;
学习如何声明函数为纯虚函数;
学习如何利用纯虚函数,编写一般成员函数。
2 实验内容
问题描述
有一家叫“周大框”的公司,用金属线为客户定制各种图案的框架。目前,该公司只生产圆(Circle)、长方形(Rectangle)和直角三角形(Right Triangle)等三种框架。
店铺负责接受客户的订单。客户除了选择图案的种类外,还要给出图案的大小,其中包括:圆形的半径;长方形的长、宽;直角三角形的两个直角边的长度。单位为厘米。
问题要求
设计类的层次。设计一个Shape抽象类;并设计Shape类的派生类:圆Circle类、长方形Rectangle类、直角三角形RightTriangle类。并且周长(Circumference)不能作为类的成员变量,而是通过调用对象的getCircumference成员函数求得。请考虑如何通过继承,尽可能多的复用代码。
选择框架。设计一个函数,生成框架产品的目录清单。用户可以重复选择多种框架图案,并给出其大小参数,直至输入-1后,选择产品结束。
框架排序。将所选框架保存在一个动态数组中(课本第9章9.2.2小节中介绍的直接访问群体——数组类,Array.h的源代码在网盘上)。在此过程中,使用插入排序算法,将所选的多种产品按周长从大到小进行排序。要求对大于(>)运算符进行重载,并在排序中使用。
结果输出
按照周长从大到小的顺序,输出以下信息。
序号.框架名称,参数,周长。例如:
1.圆,半径5,周长31.40
2.三角形,直角边3,直角边4,周长12
提示:Shape类中要包含getCircumference和showInf两个纯虚函数。
getCircumference用于计算不同图形的周长,showInf用于显示不同图形的信息。
利用基类Shape类型的动态数组存储派生类对象。借助虚函数的特性,利用基类指针访问派生类中相对应的getCircumference和showInf函数。
3 代码
#include"Array.h"
#include<iostream>
#include<iomanip>
#include<algorithm>
using namespace std;
class Shape {
public:
double virtual getCircumference() = 0;
void virtual showInf() = 0;
};
class Circle : public Shape {
private:
double radius;
public:
Circle(double r) : radius(r) {}
double virtual getCircumference() { return radius * 2 * 3.14; }
void virtual showInf() {
cout << "圆,半径" << setiosflags(2) << radius << ",周长" << setiosflags(2) << getCircumference() << endl;
}
};
class RightTriangle : public Shape {
private:
double side_one;
double side_two;
public:
RightTriangle(double o, double t) : side_one(o), side_two(t) {}
double virtual getCircumference() { return side_one + side_two + sqrt(side_one * side_one + side_two * side_two); }
void virtual showInf() {
cout << "三角形,直角边" << setiosflags(2) << side_one << ",直角边" << setiosflags(2) << side_two << ",周长"
<< setiosflags(2) << getCircumference() << endl;
}
};
class Rectangle : public Shape {
private:
double side_one;
double side_two;
public:
Rectangle(double o, double t) : side_one(o), side_two(t) {}
double virtual getCircumference() { return (side_one + side_two) * 2; }
void virtual showInf() {
cout << "长方形,长" << setiosflags(2) << side_one << ",宽" << setiosflags(2) << side_two << ",周长" << setiosflags(2)
<< getCircumference() << endl;
}
};
bool operator<(Shape &one, Shape &two) { return one.getCircumference() < two.getCircumference(); }
int main() {
Array < Shape * > parray;
int ck = 1;
int num = 0;
while (ck != -1) {
cout << "\n请输入相应数字,确定制造图案的类型:\n";
cout << "1、圆\t2、长方形\t3、直角三角形\t-1、结束\n";
cin >> ck;
switch (ck) {
case 1: {
double r;
cout << "请输入半径:";
cin >> r;
if (r <= 0) {
cout << "请输入正确的半径!";
break;
}
if (parray.getSize() <= num - 1) { parray.resize(parray.getSize() * 2); }
parray[num++] = new Circle(r);
cout << "制造成功!\n";
break;
}
case 2: {
double z1, z2;
cout << "请输入长方形的两个边长:";
cin >> z1 >> z2;
if (z1 <= 0 || z2 <= 0) {
cout << "请输入正确的边长!";
break;
}
double tt = min(z1, z2);
if (z1 < z2) {
z1 = z2;
z2 = tt;
}
Rectangle rec(z1, z2);
if (parray.getSize() <= num - 1) { parray.resize(parray.getSize() * 2); }
parray[num++] = &rec;
cout << "制造成功!\n";
break;
}
case 3: {
double b1, b2;
cout << "请输入直角三角形的两个边长:";
cin >> b1 >> b2;
if (b1 <= 0 || b2 <= 0) {
cout << "请输入正确的边长!";
break;
}
double ttt = min(b1, b2);
if (b1 < b2) {
b1 = b2;
b2 = ttt;
}
RightTriangle rig(b1, b2);
if (parray.getSize() <= num - 1) { parray.resize(parray.getSize() * 2); }
parray[num++] = &rig;
cout << "制造成功!\n";
break;
}
default:
cout << "请输入正确的操作数字!\n";
break;
}
}
for (int i = 1; i < num; i++) {
Shape *p = parray[i];
int j;
for (j = i - 1; j >= 0 && *parray[j] < *p; j--) { parray[j + 1] = parray[j]; }
parray[j + 1] = p;
}
cout << endl;
for (int i = 0; i < num; i++) { parray[i]->showInf(); }
}