免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 5061 | 回复: 7
打印 上一主题 下一主题

[C++] 基类指针和派生类指针相互赋值的原理是啥? [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2010-01-14 13:42 |只看该作者 |倒序浏览
#include <iostream>
using namespace std;
class A
{
public:
    virtual void  sprint()
{
    cout<<"A"<<endl;
}
A()
{
}

void sayhiA()
{
     cout<<"say hi from A"<<endl;
}
};

class B:  public A
{
public:
B()
{

}
void sprint()
{
    cout<<"B"<<endl;
}
void sayhiB()
{
    cout<<"say hi from B"<<endl;
}
};
int main() {

    //例程1:派生类指针赋值给基类指针

    A* a = new A();
    B* b = new B();
    cout<<"address of a ="<<a<<endl;
    cout<<"address of b ="<<b<<endl;

    a = b;
    cout<<"address of a ="<<a<<endl;
    cout<<"address of b ="<<b<<endl;
    a->sayhiA();
   
//a->sayhiB();//error

    a->sprint();

    //例程2:基类指针赋值给派生类指针

    A* a1 = new A();
    B* b1 = new B();
    cout<<"address of a1 ="<<a1<<endl;
    cout<<"address of b1 ="<<b1<<endl;

    b1 = (B*)a1;

    cout<<"address of a1 ="<<a1<<endl;
    cout<<"address of b1 ="<<b1<<endl;
    b1->sayhiA();
    b1->sayhiB();
    b1->sprint();
    return 0;
}


a=b;后,a的地址和b相同,a实际已经指向b所指向的内存块,为啥a->sayhiB();//error不被准许?a指向的不正是new B()吗?

b1 = (B*)a1; b1和a1相同,b1实际指向的应该是a1所指的地址才对。为啥b1能访问到A类所没有的sayhiB,即b1->sayhiB(); b1不是指向new A()了吗?为什么还能访问到B的内容?

求解...

论坛徽章:
0
2 [报告]
发表于 2010-01-14 13:44 |只看该作者
我机器上的运行结果
address of a =0x710218
address of b =0x710228
address of a =0x710228
address of b =0x710228
say hi from A
B
address of a1 =0x721a38
address of b1 =0x721a48
address of a1 =0x721a38
address of b1 =0x721a38
say hi from A
say hi from B
A

论坛徽章:
9
摩羯座
日期:2013-08-15 15:18:48狮子座
日期:2013-09-12 18:07:47金牛座
日期:2013-09-16 13:23:09辰龙
日期:2013-10-09 09:03:27白羊座
日期:2013-10-17 13:32:44子鼠
日期:2014-04-23 15:09:38戌狗
日期:2014-09-17 11:37:542015年亚洲杯之韩国
日期:2015-03-26 10:16:442015亚冠之武里南联
日期:2015-08-18 14:55:52
3 [报告]
发表于 2010-01-14 13:49 |只看该作者
因为编译器比较傻,它不像你这么聪明,能够直接知道这内存是谁的。
它所能知道的,那就是你对它的类型声明,它只会从这个类型中找相应的函数。

另外如果有一个函数是void func(A* a),在这个func里面,即便你再聪明,你也不会知道这个参数A* a的内存倒是A还是B,那么你希望傻傻的编译器怎么去做?

论坛徽章:
0
4 [报告]
发表于 2010-01-14 14:00 |只看该作者
原帖由 w_anthony 于 2010-1-14 13:49 发表
因为编译器比较傻,它不像你这么聪明,能够直接知道这内存是谁的。
它所能知道的,那就是你对它的类型声明,它只会从这个类型中找相应的函数。

另外如果有一个函数是void func(A* a),在这个func里面,即便 ...

你就默许是A,因为其类型申明为A,如果要用B中的函数,只有虚函数才可以。

可是如果void func(B* b),而实际上b确实个A类型,咋办?编译器还是以为b是个B类型呢(B:public A,其中B继承之A)?

[ 本帖最后由 smartvessel 于 2010-1-14 14:02 编辑 ]

论坛徽章:
9
摩羯座
日期:2013-08-15 15:18:48狮子座
日期:2013-09-12 18:07:47金牛座
日期:2013-09-16 13:23:09辰龙
日期:2013-10-09 09:03:27白羊座
日期:2013-10-17 13:32:44子鼠
日期:2014-04-23 15:09:38戌狗
日期:2014-09-17 11:37:542015年亚洲杯之韩国
日期:2015-03-26 10:16:442015亚冠之武里南联
日期:2015-08-18 14:55:52
5 [报告]
发表于 2010-01-14 15:48 |只看该作者
原帖由 smartvessel 于 2010-1-14 14:00 发表

你就默许是A,因为其类型申明为A,如果要用B中的函数,只有虚函数才可以。

可是如果void func(B* b),而实际上b确实个A类型,咋办?编译器还是以为b是个B类型呢(B:public A,其中B继承之A)?


你只要把编译器想成一个傻瓜就行了,既然是B* b,编译器只会把它当作B处理,不管它是A是B还是毫不相关的C、D。

论坛徽章:
0
6 [报告]
发表于 2010-01-14 17:14 |只看该作者

回复 #5 w_anthony 的帖子

确实是这样的
但是
原帖中
b1已经指向A类的地址,编译器虽然仍然傻瓜的讲b1认为是B类型,但是
不过b1->sayhiB();能正常执行结果,令我理解。
一个指向A类型的指针(b1虽然定义为B型指针,但它指向A类型对象),如何能找到B类型的函数?

论坛徽章:
9
摩羯座
日期:2013-08-15 15:18:48狮子座
日期:2013-09-12 18:07:47金牛座
日期:2013-09-16 13:23:09辰龙
日期:2013-10-09 09:03:27白羊座
日期:2013-10-17 13:32:44子鼠
日期:2014-04-23 15:09:38戌狗
日期:2014-09-17 11:37:542015年亚洲杯之韩国
日期:2015-03-26 10:16:442015亚冠之武里南联
日期:2015-08-18 14:55:52
7 [报告]
发表于 2010-01-14 23:33 |只看该作者
你应该把编译器想成一个傻瓜,程序运行后内存到底是谁的,这么复杂的问题连你都未必清楚,编译器又怎么会知道?
对于编译器而言,函数地址不是根据对象内存找到的,而只是根据其类型找到的,因为类型这东西一眼就能看出来,完全傻瓜式。只有虚函数例外,因为它的地址是从对象内存头四个字节的虚函数表中找到的,子类继承父类后会将这四个字节定位到子类的虚函数表中,所以才能把复杂的问题傻瓜化。
另外,你的代码中没有访问B相关的成员,不出错很正常。

论坛徽章:
0
8 [报告]
发表于 2010-01-15 09:46 |只看该作者

回复 #7 w_anthony 的帖子

thanks:wink:
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP