【多态性】C8-3 三角形还是长方形?(重难点)

题目描述
在多态概念中,基类的指针既可以指向基类的对象,又可以指向派生类的对象。我们可以使用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的用法,博大精深,这只是浅谈,希望以后还有机会可以用到

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值