基类指针指向派生类对象(C++,虚函数,内存泄漏)

说法不一,基类也叫父类,派生类也叫子类,标题也可理解为父类指针指向子类对象。这个知识点算是基础了,但是每次笔试面试都能见到,这不刚刚笔试的时候又遇到了,别说还有点忘了不是很敢确定,虽然结束之后写了一下发现没记错,还是决定记录一下。

这个知识点可以回答:

(1)什么是内存泄漏,为什么会发生,怎么解决?

(2)析构函数为什么最好是虚函数?

(3)解释一下虚函数的作用?

(4)解释一下智能指针的作用?(自动回收指针,智能指针就是为了解决这种问题的)

首先说一下创建派生类的时候是先执行基类的构造函数,然后才是派生类的构造函数。可以理解为有基础才有扩展,有父才有子。

也就是说构造函数的顺序是:基类,派生类。析构函数相反就不多说了。

直接上代码,基类指向派生类:

#include <iostream>
using namespace std;

class Base{//基类 
	public://默认是private 
		Base(){cout<<"Base begin"<<endl;}//构造函数 
		~Base(){cout<<"Base end"<<endl;}//析构函数 
		void FuncA(){cout<<"Base FuncA"<<endl;} 
		virtual void FuncB(){cout<<"Base FuncB"<<endl;}//虚函数 
};

class Child:Base{//派生类 
	public:
		Child(){cout<<"Child begin"<<endl;a=new char[10];}//注意这里添加了a指针 
		~Child(){cout<<"Child end"<<endl;delete a;}//析构释放a指针 
		void FuncA(){cout<<"Child FuncA"<<endl;}
		void FuncB(){cout<<"Child FuncB"<<endl;}//注意此处的FuncB默认的也是虚函数,平常为了可读性最好还是写上virutal 
	private:
		char *a; 
};

int main(){
	Base* p=reinterpret_cast<Base *>(new Child);//基类指针指向派生类,强制转换,否则会报错 
	//此时执行了构造函数,先"Base begin"再"Child begin" 
	p->FuncA();//执行基类的FuncA,因为指针是基类指针,执行派生类对象中的基类FuncA 
	p->FuncB();//执行派生类的FuncB,因为虚函数会找到相应对象的对应虚函数,无关是什么指针,只看对象类型 
	delete p;//注意!!!此处只执行了基类的析构函数!!!a指针没有被释放,内存泄漏了!!! 
	return 0;
}

执行结果为:

Base begin
Child begin
Base FuncA
Child FuncB
Base end

需要注意的是,这里并没有执行派生类的析构函数!!!而派生类的构造创建了指针,指针没有被释放,造成了内存泄漏!!!直到程序结束前,a指针都会一直占用着资源

怎么解决这个问题非常简单,只需要把析构函数改成虚函数就可以了

virtual ~Base(){cout<<"Base end"<<endl;}//析构函数最好都为虚函数 
Base begin
Child begin
Base FuncA
Child FuncB
Child end//派生类的析构函数
Base end

  • 0
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
以下是题目要求的基类“Role”和派生“Employee”的代码实现: ```c++ #include <iostream> #include <string> using namespace std; class Role { protected: string name; int age; char gender; public: Role() {} Role(const string &n, int a, char g) : name(n), age(a), gender(g) {} virtual ~Role() {} void setName(const string &n) { name = n; } void setAge(int a) { age = a; } void setGender(char g) { gender = g; } string getName() const { return name; } int getAge() const { return age; } char getGender() const { return gender; } virtual void display() { cout << "Name: " << name << endl; cout << "Age: " << age << endl; cout << "Gender: " << gender << endl; } }; class Employee : public Role { private: int ID; double salary; public: Employee() {} Employee(const string &n, int a, char g, int id, double s) : Role(n, a, g), ID(id), salary(s) {} Employee(const string &n, int a, char g) : Role(n, a, g), ID(0), salary(0) {} virtual ~Employee() {} void setID(int id) { ID = id; } void setSalary(double s) { salary = s; } int getID() const { return ID; } double getSalary() const { return salary; } virtual void display() { Role::display(); cout << "ID: " << ID << endl; cout << "Salary: " << salary << endl; } }; int main() { Role *p = new Employee("Tom", 25, 'M', 10001, 5000.0); p->display(); delete p; return 0; } ``` 在main函数中,我们定义了一个基类指针p,初始化为指向一个Employee对象。然后通过该基类指针调用虚函数display(),实现了多态性的效果。最后释放了p指向的内存,避免内存泄漏

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值