造成内存泄露的几种原因 C++

先考虑一种情况,对一个已知对象进行拷贝,编译系统会自动调用一种构造函数——拷贝构造函数,如果用户未定义拷贝构造函数,则会调用默认拷贝构造函数

复制代码
//main.cpp

#include <iostream>
#include "student.h"
int main()
{
         Student s1;
         Student s2(s1);//Student s2 = s1;//复制对象

         return 0;
}
复制代码
复制代码
//student.h

#ifndef STUDENT_H
#define STUDENT_H
class Student
{
         private:
               int num;
               char *name;
         public:
                Student();
                ~Student();
};

#endif
复制代码
复制代码
//student.cpp

#include "student.h"
#include <iostream>
using namespace std;

Student::Student()
{
       name = new char(20);
       cout << "Student" << endl;

}
Student::~Student()
{
        cout << "~Student " << (int)name << endl;
        delete name;
        name = NULL;
}
复制代码

执行结果:调用一次构造函数,调用两次析构函数,两个对象的指针成员所指内存相同,这会导致什么问题呢?

 

name指针被分配一次内存,但是程序结束时该内存却被释放了两次,会造成内存泄漏问题!

这是由于编译系统在我们没有自己定义拷贝构造函数时,会在拷贝对象时调用默认拷贝构造函数,进行的是浅拷贝!即对指针name拷贝后会出现两个指针指向同一个内存空间。

 

所以,在对含有指针成员的对象进行拷贝时,必须要自己定义拷贝构造函数,使拷贝后的对象指针成员有自己的内存空间,即进行深拷贝,这样就避免了内存泄漏发生。

 自己定义拷贝构造函数:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
//student.h
 
#ifndef STUDENT_H
#define STUDENT_H
class  Student
{
        private :
              int  num;
              char  *name;
       public :
               Student(); //构造函数
                ~Student(); //析构函数
                Student( const  Student &s); //拷贝构造函数,const防止对象被改变
};
 
#endif

  

复制代码
//student.cpp

#include "student.h"
#include <iostream>
#include <string.h>
using namespace std;

Student::Student()
{
      name = new char(20);
      cout << "Student " << endl;
}

Student::~Student()
{
         cout << "~Student " << (int)name << endl;
         delete name;
         name = NULL;
}

Student::Student(const Student &s)
{
         name = new char(20);
         memcpy(name, s.name, strlen(s.name));
         cout << "copy Student " << endl;
}
复制代码

执行结果:调用一次构造函数,一次自定义拷贝构造函数,两次析构函数。两个对象的指针成员所指内存不同。

 

总结:浅拷贝只是对指针的拷贝,拷贝后两个指针指向同一个内存空间,深拷贝不但对指针进行拷贝,而且对指针指向的内容进行拷贝,经深拷贝后的指针是指向两个不同地址的指针。

再说几句:

当对象中存在指针成员时,除了在复制对象时需要考虑自定义拷贝构造函数,还应该考虑以下两种情形:

1.当函数的参数为对象时,实参传递给形参的实际上是实参的一个拷贝对象,系统自动通过拷贝构造函数实现;

2.当函数的返回值为一个对象时,该对象实际上是函数内对象的一个拷贝,用于返回函数调用处。

 




1. 在类的构造函数和析构函数中没有匹配的调用newdelete函数

两种情况下会出现这种内存泄露:一是在堆里创建了对象占用了内存,但是没有显示地释放对象占用的内存;二是在类的构造函数中动态的分配了内存,但是在析构函数中没有释放内存或者没有正确的释放内存

2. 没有正确地清除嵌套的对象指针

3. 在释放对象数组时在delete中没有使用方括号

方括号是告诉编译器这个指针指向的是一个对象数组,同时也告诉编译器正确的对象地址值并调用对象的析构函数,如果没有方括号,那么这个指针就被默认为只指向一个对象,对象数组中的其他对象的析构函数就不会被调用,结果造成了内存泄露。如果在方括号中间放了一个比对象数组大小还大的数字,那么编译器就会调用无效对象(内存溢出)的析构函数,会造成堆的奔溃。如果方括号中间的数字值比对象数组的大小小的话,编译器就不能调用足够多个析构函数,结果会造成内存泄露。

释放单个对象、单个基本数据类型的变量或者是基本数据类型的数组不需要大小参数,释放定义了析构函数的对象数组才需要大小参数

4. 指向对象的指针数组不等同于对象数组

对象数组是指:数组中存放的是对象,只需要delete []p,即可调用对象数组中的每个对象的析构函数释放空间

指向对象的指针数组是指:数组中存放的是指向对象的指针,不仅要释放每个对象的空间,还要释放每个指针的空间,delete []p只是释放了每个指针,但是并没有释放对象的空间,正确的做法,是通过一个循环,将每个对象释放了,然后再把指针释放了。

5. 缺少拷贝构造函数

两次释放相同的内存是一种错误的做法,同时可能会造成堆的奔溃。

按值传递会调用(拷贝)构造函数,引用传递不会调用

C++中,如果没有定义拷贝构造函数,那么编译器就会调用默认的拷贝构造函数,会逐个成员拷贝的方式来复制数据成员,如果是以逐个成员拷贝的方式来复制指针被定义为将一个变量的地址赋给另一个变量。这种隐式的指针复制结果就是两个对象拥有指向同一个动态分配的内存空间的指针。当释放第一个对象的时候,它的析构函数就会释放与该对象有关的动态分配的内存空间。而释放第二个对象的时候,它的析构函数会释放相同的内存,这样是错误的。

所以,如果一个类里面有指针成员变量,要么必须显示的写拷贝构造函数和重载赋值运算符,要么禁用拷贝构造函数和重载赋值运算符

 C++中构造函数,拷贝构造函数和赋值函数的区别和实现参见:http://www.cnblogs.com/liushui-sky/p/7728902.html

6. 缺少重载赋值运算符

这种问题跟上述问题类似,也是逐个成员拷贝的方式复制对象,如果这个类的大小是可变的,那么结果就是造成内存泄露,如下图:

 

7. 关于nonmodifying运算符重载的常见迷思

a. 返回栈上对象的引用或者指针(也即返回局部对象的引用或者指针)。导致最后返回的是一个空引用或者空指针,因此变成野指针

b. 返回内部静态对象的引用。

c. 返回一个泄露内存的动态分配的对象。导致内存泄露,并且无法回收

解决这一类问题的办法是重载运算符函数的返回值不是类型的引用,二应该是类型的返回值,即不是 int&而是int

8. 没有将基类的析构函数定义为虚函数

当基类指针指向子类对象时,如果基类的析构函数不是virtual,那么子类的析构函数将不会被调用,子类的资源没有正确是释放,因此造成内存泄露

 

野指针:指向被释放的或者访问受限内存的指针。

造成野指针的原因:

  1. 指针变量没有被初始化(如果值不定,可以初始化为NULL
  2. 指针被free或者delete后,没有置为NULL, freedelete只是把指针所指向的内存给释放掉,并没有把指针本身干掉,此时指针指向的是“垃圾”内存。释放后的指针应该被置为NULL.
  3. 指针操作超越了变量的作用范围,比如返回指向栈内存的指针就是野指针。
  • 13
    点赞
  • 67
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 在 C/C++ 程序中,内存泄漏通常是由于程序员在使用动态内存分配函数(如 malloc、calloc、realloc)时疏忽或错误导致的。 一种常见的情况是程序员在使用动态内存分配函数申请内存后,忘记调用相应的内存释放函数(如 free)释放内存。如果这种情况发生了多次,那么会导致系统内存的浪费,并可能导致程序崩溃。 另一种情况是,程序员调用了内存释放函数,但是在调用之后仍然继续使用了已经释放的内存区域。这种情况可能会导致程序崩溃或数据错误。 此外,程序员可能会在释放指针时忘记将指针赋值为 NULL,导致内存泄漏。 在 C++ 中,还有另一种情况,即程序员使用了 new 关键字动态分配内存,但忘记使用 delete 关键字释放内存。 总的来说,内存泄漏的原因很多,但大多数情况都是由于程序员在使用动态内存分配函数时疏忽或错误导致的。程序员应该牢记内存管理的原则,尽量减少内存泄漏的发生。 ### 回答2: C/C++中可能导致内存泄漏的情况有几种: 1. 未释放堆内存:在使用`malloc()`、`new`等函数动态分配内存时,必须使用`free()`、`delete`等函数释放内存。如果忘记释放内存或者释放的次数不正确,就会出现内存泄漏的问题。 2. 循环引用:在使用引用计数的方式管理内存时,如果出现循环引用情况即两个对象相互引用,而没有外部引用指向它们,引用计数就无法减到0,导致内存泄漏。 3. 未关闭文件句柄:在使用文件操作函数`fopen()`、`open()`等打开文件时,需要通过`fclose()`、`close()`等函数来关闭文件句柄。如果忘记关闭文件句柄,系统资源将无法释放,造成内存泄漏。 4. 未释放系统资源:除了内存和文件句柄之外,还有其他系统资源也需要手动释放,如数据库连接、网络连接等。如果在使用完这些资源后没有正确释放它们,就会导致内存泄漏。 5. 堆栈不匹配:在使用C/C++的堆栈内存时,需要确保每次`malloc()`或`new`的内存分配与`free()`或`delete`的内存释放是匹配的,否则会发生内存泄漏。 6. 重复分配内存:如果在已有指针变量上多次调用`malloc()`或`new`,而没有对之前分配的内存进行释放,就会导致内存泄漏。 以上是一些常见的C/C++导致内存泄漏的情况,正确管理内存和资源是保证程序运行稳定性和性能的重要一环,开发者需要注意避免这些问题的发生。 ### 回答3: 在C/C++程序开发中,存在几种常见的情况会导致内存泄漏: 1. 动态内存分配没有被正确释放:如果在程序中使用malloc、new等方法分配内存,但是忘记释放对应的内存,则会造成内存泄漏。例如,如果在一个循环中重复分配内存但没有释放,最终会耗尽系统内存。 2. 对象生命周期没有被正确管理:在C++中,如果对象的析构函数中没有正确释放申请的资源(如内存、文件、数据库连接等),则会导致内存泄漏。这通常发生在没有及时调用对象的析构函数或者程序逻辑错误导致无法调用析构函数的情况。 3. 全局变量未释放:全局变量会在程序运行期间一直存在,如果在全局变量中分配了动态内存但未释放,那么这部分内存会一直被占用而无法回收,导致内存泄漏。因此,在使用全局变量时,需要注意释放对应的资源。 4. 异常情况未被处理:如果程序存在异常情况,但没有正确处理,导致跳过了内存释放的代码段,就会导致内存泄漏。例如,try-catch块内没有对内存进行释放操作。 5. 循环引用导致内存泄漏:在使用动态内存分配时,如果存在循环引用(两个或多个对象相互引用且没有其他对象引用它们),并且没有采用有效的内存释放策略,就会导致内存泄漏。这种情况下需要特别注意对象的析构函数中释放相关的资源。 总之,当程序分配了内存资源但没有及时释放,或者释放不完全时,就会造成内存泄漏。为了避免内存泄漏,需要在程序中正确管理内存的申请和释放,及时释放不再使用的内存
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值