C++之纯虚函数和抽象类
内容总结:
一.纯虚函数和抽象类
二.虚析构和纯虚析构
1.纯虚函数和抽象类
语法:
virtual 返回值类型 函数名 (参数列表)=0;
该类也称为抽象类。
2.抽象类特点
1.无法实例化对象
2.子类必须重写抽象类中的纯虚函数,否则也属于抽象类。
#include<iostream>
using namespace std;
//无法实例化
class Base {
public:
virtual void Fun() = 0;
};
class Son :public Base
{
public:
virtual void Fun()
{
cout << "子类中的函数" << endl;
}
};
//子类需要重写纯虚函数,不然也是抽象类
void test(int a)
{
Base *abc = new Son;
abc->Fun();
}
int main()
{
test(1);
system("pause");
return 0;
}
二.虚析构和纯虚析构
特点:
1.可以解决父类指针释放子类对象的问题
2.两者使用时,都需要具体的函数实现。
3.如果是纯虚析构,该类为抽象类,无法实例化对象。
场景:多态使用的过程中,如果子类的数据开辟到了堆区,父类指针在释放时,无法调用子类的析构代码。造成内存泄漏。
example:
#include <iostream>
#include <string>
using namespace std;
class Animal
{
public:
Animal()
{
cout << "基类的构造函数" << endl;
}
~Animal()
{
cout << "基类的析构函数" << endl;
}
virtual void Speak() = 0;
};
class Cat :public Animal
{
public:
Cat(string name)
{
cout << "子类的构造函数" << endl;
m_name = new string(name);
}
~Cat()
{
cout << "子类的析构函数" << endl;
}
void Speak()
{
cout<< *m_name << "小猫在喵喵:" << endl;
}
string *m_name;
};
void test()
{
Animal *animal = new Cat("Tom");
animal->Speak();
if (animal != NULL)
{
delete animal;
animal = NULL;
}
}
int main()
{
test();
system("pause");
return 0;
}
上述结果显示,子类的析构函数并没有发生作用,该情况下会导致内存泄漏。解决办法为将基类中的析构函数变为纯虚函数
class Animal
{
public:
Animal()
{
cout << "基类的构造函数" << endl;
}
//添加关键字virtual
virtual~Animal()
{
cout << "基类的析构函数" << endl;
}
virtual void Speak() = 0;
};
上述结果显示,子类的析构函数得到调用。堆区的内存得到回收。