dynamic_cast为什么可以做“变性手术”?

hankcs 2013-11-19 10:24:47
dynamic_cast为何可以将一个派生类指针转为另一个派生类指针并且调用另一个派生类的方法?
#include <iostream>
using namespace std;

class People
{
public:
People()
{
cout << "People constructed" << endl;
}
virtual void speak() = 0;
};

class Man:public People
{
public:
void speak()
{
cout << "A man speaks" << endl;
}

void fight()
{
cout << "A man fights" << endl;
}
};

class Woman:public People
{
private:
Man son;
public:
void speak()
{
cout << "A woman speaks" << endl;
}

void cook()
{
cout << "A woman cooks" << endl;
}

Man getSon()
{
cout << "A woman's son:" << endl;
return son;
}
};

///////////////////////////SubMain//////////////////////////////////
int main(int argc, char *argv[])
{
People *p = new Man();
p->speak();
Man *m = dynamic_cast<Man *>(p);
m->fight();
Man *m2 = (Man *)p;
m2->fight();

try
{
Woman *w = dynamic_cast<Woman *>(p);
// if (w != NULL)
// {
// w->cook();
// w->getSon().speak();
// }
// 如果不检查直接用的话,居然也可以
w->cook();
w->getSon().speak();
}
catch(bad_cast &e)
{
cout << "bad_cast exception" << endl;
}

delete p;
system("pause");
return 0;
}
///////////////////////////End Sub//////////////////////////////////
/************************************************************************/
/* output:
People constructed
A man speaks
A man fights
A man fights
A woman cooks
A woman's son:
A man speaks
*/
/************************************************************************/


我故意不检查转换是否成功,按照道理来讲w->cook();就应该报运行时错误吧?
...全文
155 4 打赏 收藏 转发到动态 举报
写回复
用AI写文章
4 条回复
切换为时间正序
请发表友善的回复…
发表回复
hankcs 2013-11-19
  • 打赏
  • 举报
回复
果然如此,我的测试工作没做好。原来抛异常还跟引用有关,谢谢
------------------------------------------------------AutoCSDN签名档------------------------------------------------------
码农场——码农播种代码、放牧思想的农场。
引用 1 楼 akirya 的回复:
w已经是空置针了 cook和getSon都不需要访问成员,所以这么写不报错。
class Woman:public People
{
private:
    Man son;
int v;
public:
    void speak()
    {
        cout << "A woman speaks" << endl;
    }
 
    void cook()
    {
        cout << "A woman cooks" << endl;
		v=0;
    }
 
    Man getSon()
    {
        cout << "A woman's son:" << endl;
        return son;
    }
};
这么写执行cook的时候就报错了。 指针类型的dynamic_cast转失败了不会抛出异常,dynamic_cast转引用类型失败才会抛出异常。
dllg1988 2013-11-19
  • 打赏
  • 举报
回复
跟dynamic_cast无关吧。我这样一样可以调用成功 Woman *w = (Woman *)0;
  • 打赏
  • 举报
回复
w已经是空置针了 cook和getSon都不需要访问成员,所以这么写不报错。
class Woman:public People
{
private:
    Man son;
int v;
public:
    void speak()
    {
        cout << "A woman speaks" << endl;
    }
 
    void cook()
    {
        cout << "A woman cooks" << endl;
		v=0;
    }
 
    Man getSon()
    {
        cout << "A woman's son:" << endl;
        return son;
    }
};
这么写执行cook的时候就报错了。 指针类型的dynamic_cast转失败了不会抛出异常,dynamic_cast转引用类型失败才会抛出异常。
96掌门师兄 2013-11-19
  • 打赏
  • 举报
回复
这个是因为虚函数的原理,做了强制转换的时候就将子对象的虚函数表拷贝过去了,所以this->fun()的时候调用的是子对象的成员方法。 不知道dynamic_cast和强制转换有什么区别

64,654

社区成员

发帖
与我相关
我的任务
社区描述
C++ 语言相关问题讨论,技术干货分享,前沿动态等
c++ 技术论坛(原bbs)
社区管理员
  • C++ 语言社区
  • encoderlee
  • paschen
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
  1. 请不要发布与C++技术无关的贴子
  2. 请不要发布与技术无关的招聘、广告的帖子
  3. 请尽可能的描述清楚你的问题,如果涉及到代码请尽可能的格式化一下

试试用AI创作助手写篇文章吧