C++关于一个函数中new内存泄露的列子

简介: 首先明白几个基础 1、函数按值传递和按值返回的时候都会调用复制构造函数 2、一般在函数体内定义的栈变量是不能返回其地址或者引用给主调函数的,因为在函数结束的时候这些栈变量将释放 3、可以使用new的方式建立堆内存的方式,然后返回引用或者指针,因为new这种方式建立的堆内存并不随函数的结束而结束,      而指针变量释放但是指针本生的值已经返回。
首先明白几个基础
1、函数按值传递和按值返回的时候都会调用复制构造函数
2、一般在函数体内定义的栈变量是不能返回其地址或者引用给主调函数的,因为在函数结束的时候这些栈变量将释放
3、可以使用new的方式建立堆内存的方式,然后返回引用或者指针,因为new这种方式建立的堆内存并不随函数的结束而结束,
     而指针变量释放但是指针本生的值已经返回。同时也可以按值放回,但是这种情况下将可能出现内存泄露
来看下面的代码

点击(此处)折叠或打开

  1. /*************************************************************************
  2.     > File Name: testcc.cpp
  3.     > Author: gaopeng
  4.     > Mail: gaopp_200217@163.com
  5.     > Created Time: Thu 01 Sep 2016 09:06:53 PM CST
  6.  ************************************************************************/

  7. #include<iostream>
  8. using namespace std;


  9. class testa
  10. {
  11.         private:
  12.                 int i;
  13.         public:
  14.                 testa(const int m){
  15.                         cout<<"create a object\n";
  16.                         i=m;
  17.                 }
  18.                 const int& geti() const {
  19.                         return i;
  20.                 }
  21.                 testa(const testa& m ){
  22.                         cout<<"copy funcation\n";
  23.                         i=m.i;
  24.                 }
  25.                 ~testa(){
  26.                         cout<<"discard a object\n";
  27.                 }
  28.                 testa operator=(const testa& c)
  29.                 {
  30.                         cout<<"= funcation\n";
  31.                         i = c.i;
  32.                 }

  33. };

  34. testa func()
  35. {
  36.         cout<<"in func function\n";
  37.         //testa p(10);
  38.         testa* p = new testa(1);
  39.         cout<<p<<endl;
  40.         cout<<"end func\n";
  41.         return *p;
  42. }


  43. int main(void)
  44. {
  45.         testa m = func(); //copy
  46.         cout<<&m<<endl;
  47.         cout<<m.geti()<<endl;
  48.         return 0;
  49. }
程序说明:
这里 testa *  p  =  new testa ( 1 ) ;建立一块堆内存
这里 return  * p ;按值返回,按值返回会调用复制构造函数给值赋予给新建个对象m
程序结束后调用m的析构函数,但是这里new出来的内存空间已经没有可以指向的指针
因为p已经释放,而返回的是*p,这块内存已经泄露。我们跑一下看看:
in func function   --调用func函数
create a object   --new创建的testa的堆内存  testa *  p  =  new testa ( 1 ) ;
0x1914010        --new的地址  cout < < p < < endl ;
end func           --结束func函数  cout < < "end func\n" ;
copy funcation   --按值返回调用复制构造函数,将值赋予给新的变量 m  testa m  =  func ( ) ;
0x7fffb9c438a0  --新对象m的地址 cout < < & m < < endl ;
1
discard a object  --析构函数释放栈对象m的空间

这里我们发现new的堆内存空间没有被析构,那么内存已经泄露。
那么我们怎么不大量改变程序的情况下来消除这种问题呢
当然是使用指针或者引用来返回

点击(此处)折叠或打开

  1. testa* func()
  2. {
  3.         cout<<"in func function\n";
  4.         //testa p(10);
  5.         testa* p = new testa(1);
  6.         cout<<p<<endl;
  7.         cout<<"end func\n";
  8.         return p;
  9. }


  10. int main(void)
  11. {
  12.         {
  13.                 testa* m = func(); //copy
  14.                 cout<<m<<endl;
  15.                 cout<<m->geti()<<endl;
  16.                 delete m;
  17.         }
  18.         return 0;
  19. }
这一在main中我把定义m指针到删除放到了一个block中,这样在block结束的时候就释放了m避免了空指针的存在。
下面是引用

点击(此处)折叠或打开

  1. testa& func()
  2. {
  3.         cout<<"in func function\n";
  4.         //testa p(10);
  5.         testa* p = new testa(1);
  6.         cout<<p<<endl;
  7.         cout<<"end func\n";
  8.         return *p;
  9. }


  10. int main(void)
  11. {
  12.         {
  13.                 testa& m = func(); //copy
  14.                 cout<<&m<<endl;
  15.                 cout<<m.geti()<<endl;
  16.                 delete &m;
  17.         }
  18.         return 0;
  19. }
同样main中的这个程序块是为了避免空引用
输出如下:
in func function  
create a object
0x1989010
end func
0x1989010   
1
discard a object

可以看到地址都相同,最后的析构函数是我调用delete执行的。



相关文章
|
2月前
|
C语言 C++
C语言 之 内存函数
C语言 之 内存函数
38 3
|
15天前
|
存储 缓存 算法
【C语言】内存管理函数详细讲解
在C语言编程中,内存管理是至关重要的。动态内存分配函数允许程序在运行时请求和释放内存,这对于处理不确定大小的数据结构至关重要。以下是C语言内存管理函数的详细讲解,包括每个函数的功能、标准格式、示例代码、代码解释及其输出。
47 6
|
27天前
|
存储 缓存 C语言
【c++】动态内存管理
本文介绍了C++中动态内存管理的新方式——`new`和`delete`操作符,详细探讨了它们的使用方法及与C语言中`malloc`/`free`的区别。文章首先回顾了C语言中的动态内存管理,接着通过代码实例展示了`new`和`delete`的基本用法,包括对内置类型和自定义类型的动态内存分配与释放。此外,文章还深入解析了`operator new`和`operator delete`的底层实现,以及定位new表达式的应用,最后总结了`malloc`/`free`与`new`/`delete`的主要差异。
42 3
|
1月前
|
存储 编译器 Linux
【c++】类和对象(上)(类的定义格式、访问限定符、类域、类的实例化、对象的内存大小、this指针)
本文介绍了C++中的类和对象,包括类的概念、定义格式、访问限定符、类域、对象的创建及内存大小、以及this指针。通过示例代码详细解释了类的定义、成员函数和成员变量的作用,以及如何使用访问限定符控制成员的访问权限。此外,还讨论了对象的内存分配规则和this指针的使用场景,帮助读者深入理解面向对象编程的核心概念。
82 4
|
2月前
|
程序员 C++ 容器
在 C++中,realloc 函数返回 NULL 时,需要手动释放原来的内存吗?
在 C++ 中,当 realloc 函数返回 NULL 时,表示内存重新分配失败,但原内存块仍然有效,因此需要手动释放原来的内存,以避免内存泄漏。
|
2月前
|
存储 C语言 C++
【C++打怪之路Lv6】-- 内存管理
【C++打怪之路Lv6】-- 内存管理
44 0
【C++打怪之路Lv6】-- 内存管理
|
2月前
|
存储 C语言
【c语言】字符串函数和内存函数
本文介绍了C语言中常用的字符串函数和内存函数,包括`strlen`、`strcpy`、`strcat`、`strcmp`、`strstr`、`strncpy`、`strncat`、`strncmp`、`strtok`、`memcpy`、`memmove`和`memset`等函数的使用方法及模拟实现。文章详细讲解了每个函数的功能、参数、返回值,并提供了具体的代码示例,帮助读者更好地理解和掌握这些函数的应用。
37 0
|
2月前
|
C++
C/C++内存管理(下)
C/C++内存管理(下)
51 0
|
21天前
|
存储 编译器 C语言
【c++丨STL】string类的使用
本文介绍了C++中`string`类的基本概念及其主要接口。`string`类在C++标准库中扮演着重要角色,它提供了比C语言中字符串处理函数更丰富、安全和便捷的功能。文章详细讲解了`string`类的构造函数、赋值运算符、容量管理接口、元素访问及遍历方法、字符串修改操作、字符串运算接口、常量成员和非成员函数等内容。通过实例演示了如何使用这些接口进行字符串的创建、修改、查找和比较等操作,帮助读者更好地理解和掌握`string`类的应用。
30 2
|
27天前
|
存储 编译器 C++
【c++】类和对象(下)(取地址运算符重载、深究构造函数、类型转换、static修饰成员、友元、内部类、匿名对象)
本文介绍了C++中类和对象的高级特性,包括取地址运算符重载、构造函数的初始化列表、类型转换、static修饰成员、友元、内部类及匿名对象等内容。文章详细解释了每个概念的使用方法和注意事项,帮助读者深入了解C++面向对象编程的核心机制。
70 5