父类为形参,子类为实参时的注意事项
采用直接值传递
代码示例
#include <iostream>
using namespace std;
class Fish
{
public:
virtual void ShowInf()
{
cout << "我是一条fish" << endl;
}
};
class Carp : public Fish
{
public:
void ShowInf()
{
cout << "我是一条carp" << endl;
}
};
void ShowInf(Fish fish) // 直接进行普通的值传递
{
fish.ShowInf();
}
int main()
{
Carp carp;
ShowInf(carp);
}
代码到底讲述了什么?
当函数参数是父类对象且传入的参数是子类对象,然而我们用的是普通值传递时,子类对象仅仅将自己继承的那部分数据复制给子类形参。
运行结果
采用引用值传递
代码示例
#include <iostream>
using namespace std;
class Fish
{
public:
virtual void ShowInf()
{
cout << "我是一条fish" << endl;
}
};
class Carp : public Fish
{
public:
void ShowInf()
{
cout << "我是一条carp" << endl;
}
};
void ShowInf(Fish& fish) // 进行引用值传递
{
fish.ShowInf();
}
int main()
{
Carp carp;
ShowInf(carp);
}
代码到底讲述了什么?
当用&引用的方式进行值传递时,就相当将自己阉割成父类对象的样子,即将只属于自己的那一部分割去。然后将剩下的这一部分当作实参传递给函数。
我们这里要注意的是虚函数,当父类中有与子类同名的虚函数时,此时子类的成员函数会覆盖父类中的虚成员函数。因此,阉割剩下的部分并不与原父类成员相同。虚函数的作用原理详见“虚函数表”。
虚函数表
我们看到父类中的ShowInf()函数被声明为virtual类型的了,那虚函数是如何作用的呢?
我个人理解:虚函数的虚字可以理解为“形同虚设”,当我们声明父类的ShowInf()函数为虚函数同时我们在子类中也声明一个ShowInf()函数,那么在子类中,ShowInf()函数正好可以覆盖父类中的同名函数(形同虚设的函数)。
详细讲解请参考:https://blog.csdn.net/weixin_45590473/article/details/107296026
运行结果
采用深复制进行传递
#include <iostream>
using namespace std;
class Fish
{
public:
virtual void ShowInf()
{
cout << "我是一条fish" << endl;
}
};
class Carp : public Fish
{
public:
void ShowInf()
{
cout << "我是一条carp" << endl;
}
};
void ShowInf(Fish* fish)
{
dynamic_cast<Carp*>(fish)->ShowInf(); // 进行了有继承关系类之间的强制类型转换
}
int main()
{
Carp carp;
ShowInf(&carp);
}
深复制其实传递与操作的是对象的指针,因为当我们获得地址(指针内容)后,我们可以调整指针类型用来访问不同的有效区间,有效区间仅仅指的是内存中与指针类型相同的区域,并不会像值传递的方式阉割了一部分内容,用指针作为形参既灵活(可以调整指针类型从而调整指针指向内容)又高效(传递指针比值传递快得多)。