class A{
public:
A(int a){
this->a=a;
}
int a;
};
class B:public A{
public:
B(int a,int b):A(a){
this->b=b;
}
int b;
};
A类占内存大小范围:int a; 占4个字节.
B类占内存大小范围:int a; 加上 int b;一共占8个字节.
基类指针(a)指向派生类(B):
A *a=new B(1,2);
因为a是A类型指针,所以*a只能指向A类的int a(4字节长度),如果访问int b;就越界了.
总结:
<1>.当基类指针指向派生类的时候,只能操作派生类从基类中继承过来的数据.
<2>.当派生类指向基类的指针,因为内存空间比基类长,访问的话会导致内存溢出,所以不允许派生类的指针指向基类。
加深理解:
<1>.通常来说,子类肯定含有父类所有的成员变量和方法函数.所以用父类指针指向子类时,没有问题
,因为父类有的,子类都有,不会出现非法访问问题.
<2>.如果用子类指针指向父类的话,一旦访问子类特有的方法函数或者成员变量(基类是没有的),就会出现非法访问;
因为被子类指针指向的由父类创建的对象,根本没有要访问的那些内容,那些是子类特有的,只有用子类初始化对象时才会有.
通过 static_cast、dynamic_cast可以解决这个问题
1 static_cast
用法:static_cast < type-id > ( exdivssion )
该运算符把exdivssion转换为type-id类型,但没有运行时类型检查来保证转换的安全性。它主要有如下几种用法:
①用于类层次结构中基类和子类之间指针或引用的转换。
进行上行转换(把子类的指针或引用转换成基类表示)是安全的;
进行下行转换(把基类指针或引用转换成子类表示)时,由于没有动态类型检查,所以是不安全的。
②用于基本数据类型之间的转换,如把int转换成char,把int转换成enum。这种转换的安全性也要开发人员来保证。
③把空指针转换成目标类型的空指针。
④把任何类型的表达式转换成void类型。
注意:static_cast不能转换掉exdivssion的const、volitale、或者__unaligned属性。
2 dynamic_cast
用法:dynamic_cast < type-id > ( exdivssion )
该运算符把exdivssion转换成type-id类型的对象。Type-id必须是类的指针、类的引用或者void *;
如果type-id是类指针类型,那么exdivssion也必须是一个指针,如果type-id是一个引用,那么exdivssion也必须是一个引用。
dynamic_cast主要用于类层次间的上行转换和下行转换,还可以用于类之间的交叉转换。
在类层次间进行上行转换时,dynamic_cast和static_cast的效果是一样的;
在进行下行转换时,dynamic_cast具有类型检查的功能,比static_cast更安全。
使用dynamic_cast做类型转化会提示报错 “is not polymorphic”
这是因为使用dynamic_cast时,基类为包含至少一个虚函数的基类;
#include <iostream>
#include<string>
using namespace std;
class A{
public:
A(int a){
this->a=a;
}
int a;
virtual void fun(){}; //虚函数
};
class B:public A{
public:
B(int a,int b):A(a){
this->b=b;
}
int b;
};
int main() {
A *a=new B(1,2);
(static_cast<B*>(a))->b;
if(B *b = dynamic_cast<B*>(a)){
}
}