C++深入讲解对象的销毁之析构函数

目录
  • 一、对象的销毁
  • 二、析构函数
  • 三、小结

一、对象的销毁

  • 生活中的对象都是被初始化后才上市的
  • 生活中的对象被销毁前会做一些清理工作
  • —股而言,需要销毁的对象都应该做清理

解决方案

  • 为每个类都提供一个 public 的 free 函数
  • 对象不再需要时立即调用 free 函数进行清理

如下:

存在的问题

  • free 只是一个普通的函数,必须显示的调用
  • 对象销毁前没有做清理,很可能造成资源泄漏

C++ 编译器是否能够自动调用某个特殊的函数进行对象的清理?

二、析构函数

C++ 的类中可以定义一个特殊的清理函数

  • 这个特殊的清理函数叫做析构函数
  • 析构函数的功能与构造函数相反

定义:~ClassName()

  • 析构函数没有参数也没有返回值类型声明
  • 析构函数在对象销毁时自动被调用

下面开始简单使用析构函数:

#include <stdio.h>

class Test
{
public:
    Test()
    {
        printf("Test()\n");
    }
    ~Test()
    {
        printf("~Test()\n");
    }
};

int main()
{
    Test t;

    return 0;
}

输出结果如下:

t 虽然是对象,但是本质上也是局部变量,在 return 0 之前会销毁,t 被销毁时析构函数会被自动调用。

下面再来看一个例子:

#include <stdio.h>

class Test
{
    int mi;
public:
    Test(int i)
    {
        mi = i;
        printf("Test(): %d\n", mi);
    }
    ~Test()
    {
        printf("~Test(): %d\n", mi);
    }
};

int main()
{
    Test t(1);

    Test* pt = new Test(2);

    delete pt;

    return 0;
}

输出结果如下:

析构函数的定义准则

当类中自定义了构造函数,并且构造函数中使用了系统资源(如∶内存申请,文件打开,等) ,则需要自定义析构函数。

下面再来看一个实验:

IntArray.h:

#ifndef _INTARRAY_H_
#define _INTARRAY_H_

class IntArray
{
private:
    int m_length;
    int* m_pointer;
public:
    IntArray(int len);
    IntArray(const IntArray& obj);
    int length();
    bool get(int index, int& value);
    bool set(int index ,int value);
    ~IntArray();
};

#endif

IntArray.cpp:

#include "IntArray.h"
#include "stdio.h"

IntArray::IntArray(int len)

{

    m_pointer = new int[len];

    for(int i=0; i<len; i++)

    {

        m_pointer[i] = 0;

    }

    m_length = len;

}

IntArray::IntArray(const IntArray& obj)

{

    m_length = obj.m_length;

    m_pointer = new int[obj.m_length];

    for(int i=0; i<obj.m_length; i++)

    {

        m_pointer[i] = obj.m_pointer[i];

    }

}

int IntArray::length()

{

    return m_length;

}

bool IntArray::get(int index, int& value)

{

    bool ret = (0 <= index) && (index < length());

    if( ret )

    {

        value = m_pointer[index];

    }

    return ret;

}

bool IntArray::set(int index, int value)

{

    bool ret = (0 <= index) && (index < length());

    if( ret )

    {

        m_pointer[index] = value;

    }

    return ret;

}

IntArray::~IntArray()

{
    printf("do it\n");

    delete[]m_pointer;

}

main.cpp:

#include <stdio.h>
#include "IntArray.h"

int main()
{
    IntArray a(5);    

    for(int i=0; i<a.length(); i++)
    {
        a.set(i, i + 1);
    }

    for(int i=0; i<a.length(); i++)
    {
        int value = 0;

        if( a.get(i, value) )
        {
            printf("a[%d] = %d\n", i, value);
        }
    }

    IntArray b = a;

    for(int i=0; i<b.length(); i++)
    {
        int value = 0;

        if( b.get(i, value) )
        {
            printf("b[%d] = %d\n", i, value);
        }
    }

    return 0;
}

输出结果如下:

可以看到 do it 输出两次,也就是说析构函数被自动调用两次。

三、小结

  • 析构函数是对象销毁时进行清理的特殊函数
  • 析构函数在对象销毁时自动被调用
  • 析构函数是对象释放系统资源的保障

到此这篇关于C++深入讲解对象的销毁与析构函数的文章就介绍到这了,更多相关C++ 对象的销毁内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • c++ 入门——浅析构造函数和析构函数

    前文回顾 本文档环境基于Vscode + GCC + CodeRunner 关于C++的环境搭建请参考下面链接: https://www.jb51.net/article/186542.htm 由于本人具有C#开发经验,部分相同的知识就不再赘述了.只列一下需要记录的知识. HelloWorld cout 代表输出<< cin 代表输入 >> endl;代表换行,清空缓冲区. #include <iostream> int main() { std::cout <&

  • C++编程析构函数拷贝构造函数使用示例详解

    目录 构造函数 析构函数 拷贝构造之深拷贝和浅拷贝 深浅拷贝区别 首先定义一个类进行操作. class MM { public: protected: int year; string name; } 构造函数在类中默认有一个无参的构造函数 默认的构造函数为 类名(){}:这个构造函数 如果直接写了构造函数那么这个构造函数将会没有 构造函数 class MM { public: //MM() {};//无参构造函数 MM(int year, string name) :year(year), n

  • 全面解析C++中的析构函数

    "析构函数"是构造函数的反向函数.在销毁(释放)对象时将调用它们.通过在类名前面放置一个波形符 (~) 将函数指定为类的析构函数.例如,声明 String 类的析构函数:~String(). 在 /clr 编译中,析构函数在释放托管和非托管资源方面发挥了特殊作用. 析构函数通常用于在不再需要某个对象时"清理"此对象.请考虑 String 类的以下声明: // spec1_destructors.cpp #include <string.h> class

  • 详解C++中的析构函数

    简介 析构函数(Destructors),是对象的成员函数,没有返回值也没有参数,且一个类只有一个析构函数,当对象被销毁的时候调用,被销毁通常有这么几个情况. 函数执行结束 程序执行结束 程序块包含的局部变量 delete操作 什么时候要自己写析构函数? 编译器会自动创建默认的析构函数,通常都没有问题,但是当我们在类中动态分配了内存空间时,我们需要手段的回收这块空间,防止内存溢出.就像这样 class String { private: char *s; int size; public: St

  • C++深入讲解对象的销毁之析构函数

    目录 一.对象的销毁 二.析构函数 三.小结 一.对象的销毁 生活中的对象都是被初始化后才上市的 生活中的对象被销毁前会做一些清理工作 —股而言,需要销毁的对象都应该做清理 解决方案 为每个类都提供一个 public 的 free 函数 对象不再需要时立即调用 free 函数进行清理 如下: 存在的问题 free 只是一个普通的函数,必须显示的调用 对象销毁前没有做清理,很可能造成资源泄漏 C++ 编译器是否能够自动调用某个特殊的函数进行对象的清理? 二.析构函数 C++ 的类中可以定义一个特殊

  • C++函数返回值为对象时,构造析构函数的执行细节

    看如下代码: 复制代码 代码如下: #include<iostream>class TestConstructor{public:    TestConstructor()    {        std::cout<<"TestConstructor()"<<std::endl;    }    ~TestConstructor()    {        std::cout<<"~TestConstructor()"

  • Java中对象的销毁方法分析

    本文较为详细的分析了Java中对象的销毁方法.分享给大家供大家参考.具体分析如下: Java中的基本数据类型变量和对象的名称引用变量如定义在方法中,都为局部变量.但对象本身不一定是局部生命周期.如函数外存在其他对该对象的引用变量,则该对象的生命周期延伸至该其他引用变量所在的块. 如从被调用函数参数引用传值或返回值到主调用函数所在的对象类型变量中,则该对象都仍存在(但被调用函数的该对象的引用变量生命周期结束,因此引用变量是局部变量),此时对象突破了局部变量的局部生命期. Java对象销毁 Java

  • 你真的知道Java中对象的销毁吗

    在日常的开发中.我们都知道,Java的内存清理是通过垃圾回收器进行的,那么其是如何将没用的对象被被清理掉的呢? Java 语言的内存自动回收称为垃圾回收(Garbage Collection)机制,简称 GC.垃圾回收机制是指 JVM 用于释放那些不再使用的对象所占用的内存. Java对象在使用后需要清理. 对象清理是释放该对象所占用的内存. 在创建对象时,用户必须使用new操作符为对象分配内存. 清除对象后,系统会自动回收内存,不需要用户进行额外的处理. 这也是Java语言的一个特性,它使程序

  • Java 超详细讲解对象的构造及初始化

    目录 如何初始化对象 构造方法 特性 默认初始化 就地初始化 如何初始化对象 我们知道再Java方法内部定义一个局部变量的时候,必须要初始化,否则就会编译失败 要让这串代码通过编译,很简单,只需要在正式使用a之前,给a设置一个初始值就好那么对于创造好的对象来说,我们也要进行相对应的初始化我们先写一个Mydate的类 public class MyDate { public int year; public int month; public int day; /** * 设置日期: */ pub

  • C++ 详细讲解对象的构造顺序

    目录 一.局部对象的构造顺序 二.堆对象的构造顺序 三.全局对象的构造顺序 命令行 四.小结 一.局部对象的构造顺序 对于局部对象 当程序执行流到达对象的定义语句时进行构造 下面看一个局部对象的构造示例: #include <stdio.h> class Test { private: int mi; public: Test(int i) { mi = i; printf("Test(int i): %d\n", mi); } Test(const Test& o

  • C++详细讲解对象的构造

    目录 一.对象的构造(上) 1.1 对象的初始值 1.2 对象的初始化 1.3 小结 二.对象的构造(中) 2.1 构造函数 2.2小实例 2.3 小结 三.对象的构造(下) 3.1 特殊的构造函数 3.2 拷贝构造函数 3.3 小结 一.对象的构造(上) 1.1 对象的初始值 问题:对象中成员变量的初始值是多少? 下面的类定义中成员变量 i 和 j 的初始值为多少? 下面看一段成员变量初始值的代码: #include<stdio.h> class Test { private: int i;

  • php基础知识:类与对象(3) 构造函数和析构函数

    构造函数 PHP 5 允行开发者在一个类中定义一个方法作为构造函数.具有构造函数的类会在每次创建对象时先调用此方法,所以非常适合在使用对象之前做一些初始化工作.  注意:  如果子类中定义了构造函数则不会暗中调用其父类的构造函数.要执行父类的构造函数,需要在子类的构造函数中调用 parent::__construct().(??和其他语言明显不同??) 例10.8.使用新标准的构造函数 class BaseClass {   function __construct() {       prin

  • PHP析构函数destruct与垃圾回收机制的讲解

    析构函数 当某个对象成为垃圾或者当对象被显式销毁时执行. PHP5中提供的析构函数是__destruct,其与构造方法__construct相对应. 垃圾回收--GC(Garbage Collector) 在PHP中,没有任何变量指向这个对象时,这个对象就成为垃圾,PHP会将其在内存中销毁. 这是PHP的GC(Garbage Collector)垃圾处理机制,垃圾加收可以防止内存溢出. 当一个PHP线程结束时,当前占用的所有内存空间都会被销毁,当前程序中的所有对象同样被销毁. __destruc

  • 析构函数与php的垃圾回收机制详解

    析构函数:当某个对象成为垃圾或者当对象被显式销毁时执行. GC(Garbage Collector) 在PHP中,没有任何变量指向这个对象时,这个对象就成为垃圾.PHP会将其在内存中销毁. 这是PHP的GC(Garbage Collector)垃圾处理机制,防止内存溢出. 当一个PHP线程结束时,当前占用的所有内存空间都会被销毁,当前程序中的所有对象同样被销毁. __destruct() 析构函数 __destruct() 析构函数,是在垃圾对象被回收时执行. 析构函数也可以被显式调用,但不要这

随机推荐