这篇文章主要介绍了C++析构函数怎么使用的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇C++析构函数怎么使用文章都会有所收获,下面我们一起来看看吧。
析构函数是特殊的成员函数
特征如下:
析构函数名是~类名;
无参数无返回值;
一个类有且只有一个析构函数;
对象声明周期结束,编译器自动调用析构函数;
class Stack { public: Stack(int capacity = 4) : _size(0), _capacity(capacity), _p(new int[_capacity]) { cout << "Stack(int capacity = 4)" << endl; } ~Stack() { cout << "~Stack()" << endl; if (_p) { delete[](_p); _p = nullptr; } _size = _capacity = 0; } private: int _capacity; int _size; int* _p; }; int main() { Stack s; return 0;//程序结束,调用s的析构函数 }
输出:
class String { public: String(const char* str = "songxin") { cout << "String(const char* str = \"songxin\")" << endl; _str = (char*)malloc(strlen(str) + 1); strcpy(_str, str); } ~String() { cout << "~String()" << endl; free(_str); _str = nullptr; } private: char* _str; }; class Person { public: Person() : _age(20), _name() { cout << "Person()" << endl; } ~Person() { cout << "~Person()" << endl; } private: String _name; int _age; }; int main() { Person p; return 0; }
输出:
析构函数在程序即将结束时,调用了Person的析构函数,在Person类的析构函数即将结束接着调用String类的析构函数。
归纳一下:
析构函数是与构造函数执行相反的操作的,构造函数负责给对象成员变量初始化并加载资源,而析构函数则是给对象的成员变量清理资源,而不是清理对象本身。
编译器默认生成的析构函数能做些什么工作呢?我们前面已经介绍了编译器生成的构造函数会去只会处理自定义类型的成员变量,那么析构既然和构造相对应,析构也应该是只去处理自定义类型的成员变量吧,确实如此,析构函数不会对内置类型有任何处理,只会在调用自身的析构后再去调用自定义类型成员的析构。
关于编译器自动生成的析构函数,下面的程序我们会看到,编译器生成的析构函数,会对自定类型成员调用它的析构函数。
class String { public: String(const char* str = "songxin") { cout << "String(const char* str = \"songxin\")" << endl; _str = (char*)malloc(strlen(str) + 1); strcpy(_str, str); } ~String() { cout << "~String()" << endl; free(_str); _str = nullptr; } private: char* _str; }; class Person { public: Person() : _age(20), _name() { cout << "Person()" << endl; } private: String _name; int _age; }; int main() { Person p; return 0; }
输出:
默认生成的析构函数对成员变量的处理
内置类型不处理;
自定义类型成员调用相应的析构函数;
那成员变量中的内置类型处不处理其实都无所谓嘛,反正都要归还给操作系统,但是有例外:
如果成员变量含有指针,并且指针指向一块我们正使用的空间,指针也是内置类型,那如果不释放指针指向的那块空间就会造成内存泄漏,而编译器生成的析构函数是不会处理此情况的,因为需要我们在析构函数中主动释放内存,也就是说需要我们显式的去定义析构函数。
class Stack { public: Stack(int capacity = 4) : _size(0), _capacity(capacity), _p(new int[_capacity])//使用new去申请内存 { cout << "Stack(int capacity = 4)" << endl; } ~Stack() { cout << "~Stack()" << endl; if (_p) { delete[](_p);//释放内存 _p = nullptr; } _size = _capacity = 0; } private: int _capacity; int _size; int* _p; }; int main() { Stack s; return 0;//程序结束,调用s的析构函数 }
析构函数无论是我们显式定义的还是编译器生成的,都会在对象的声明周期结束时自动调用,并且会调用自定义类型成员变量的析构函数来释放资源,而对内置类型不做处理。
可以不显式定义析构函数的情况
类的成员都是自定义类型的;
类的成员都是非指针的内置类型;
成员有指针,但并没有管理内存资源;
如果类的成员变量有指针类型,并且我们让指针指向了一块动态分配的空间,那么就需要我们自己写析构函数了。
总结:不是类直接管理另一块内存资源的,就不需要写析构函数,编译器自己生成的就能处理。
关于“C++析构函数怎么使用”这篇文章的内容就介绍到这里,感谢各位的阅读!相信大家对“C++析构函数怎么使用”知识都有一定的了解,大家如果还想学习更多知识,欢迎关注亿速云行业资讯频道。
免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。