题目描述
在多态概念中,基类的指针既可以指向基类的对象,又可以指向派生类的对象。我们可以使用dynamic_cast类型转换操作符来判断当前指针(必须是多态类型)是否能够转换成为某个目的类型的指针。
同学们先查找dynamic_cast的使用说明(如http://en.wikipedia.org/wiki/Run-time_type_information#dynamic_cast),然后使用该类型转换操作符完成下面程序(该题无输入)。
函数int getVertexCount(Shape * b)计算b的顶点数目,若b指向Shape类型,返回值为0;若b指向Triangle类型,返回值为3;若b指向Rectangle类型,返回值为4。
代码实现
#include <cstdio>
#include <cstring>
#include <iostream>
using namespace std;
class Shape {
public:
Shape() {}
virtual ~Shape() {}
};
class Triangle : public Shape {
public:
Triangle() {}
~Triangle() {}
};
class Rectangle : public Shape {
public:
Rectangle() {}
~Rectangle() {}
};
/*用dynamic_cast类型转换操作符完成该函数*/
int getVertexCount(Shape * b) {
// Shape *pa = dynamic_cast<Shape*>(b);
Triangle *pb = dynamic_cast<Triangle*>(b);
Rectangle *pc = dynamic_cast<Rectangle*>(b);
// if (pa != NULL)
// {
// return 0;
// }
if (pb != NULL)
{
return 3;
}if (pc != NULL)
{
return 4;
}
else
{
return 0;
}
}
int main() {
Shape s;
cout << getVertexCount(&s) << endl;
Triangle t;
cout << getVertexCount(&t) << endl;
Rectangle r;
cout << getVertexCount(&r) << endl;
return 0;
}
本题重难点在于dynamic_cast的使用
对于dynamic的理解参考https://blog.csdn.net/qq_41111294/article/details/81178391
https://blog.csdn.net/hongkangwl/article/details/21161713
https://blog.csdn.net/u012999985/article/details/69056556
可以了解到的重点为
将一个基类对象指针(或引用)cast到继承类指针,dynamic_cast会根据基类指针是否真正指向继承类指针来做相应处理,
对指针进行dynamic_cast,失败返回null,成功返回正常cast后的对象指针;
对引用进行dynamic_cast,失败抛出一个异常,成功返回正常cast后的对象引用。在类层次间进行上行转换时,dynamic_cast和static_cast的效果是一样的;
在进行下行转换时,dynamic_cast具有类型检查的功能,比static_cast更安全。
而真正点睛之笔在于这篇文章
https://blog.csdn.net/lc_910927/article/details/39348673
即dynamic_cast用于将基类的引用或指针转化为派生类的引用或指针。但使用条件是:1、基类的指针或引用确实绑定到派生类的对象上;2、只有当基类至少含有一个虚函数的时候才能使用dynamic_cast。原因是RTTI机制依赖于虚函数表(inside
C++ object
model第一章给出了解释),而dynamic_cast是RTTI的一种,所以必须要有虚函数表的支持,也就是要虚类中至少有一个虚函数。
再来看我们的代码
int getVertexCount(Shape * b) {
// Shape *pa = dynamic_cast<Shape*>(b);
Triangle *pb = dynamic_cast<Triangle*>(b);
Rectangle *pc = dynamic_cast<Rectangle*>(b);
// if (pa != NULL)
// {
// return 0;
// }
if (pb != NULL)
{
return 3;
}if (pc != NULL)
{
return 4;
}
else
{
return 0;
}
}
在这个函数中,传入的是一个基类的指针,所以我们只需要判断这个基类指针是否指向派生类,不用判断是否指向基类。因为传入的是基类指针,所以一定指向基类,也就是注释掉代码的错误所在。所以只需要判断是否指向了两个派生类,若没有则只指向基类。
关于dynamic的用法,博大精深,这只是浅谈,希望以后还有机会可以用到