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执行的。



目录
打赏
0
0
0
0
91
分享
相关文章
玩转C++内存管理:从新手到高手的必备指南
C++中的内存管理是编写高效、可靠程序的关键所在。C++不仅继承了C语言的内存管理方式,还增加了面向对象的内存分配机制,使得内存管理既有灵活性,也更加复杂。学习内存管理不仅有助于提升程序效率,还有助于理解计算机的工作原理和资源分配策略。
【硬核】C++11并发:内存模型和原子类型
本文从C++11并发编程中的关键概念——内存模型与原子类型入手,结合详尽的代码示例,抽丝剥茧地介绍了如何实现无锁化并发的性能优化。
206 68
彻底摘明白 C++ 的动态内存分配原理
大家好,我是V哥。C++的动态内存分配允许程序在运行时请求和释放内存,主要通过`new`/`delete`(用于对象)及`malloc`/`calloc`/`realloc`/`free`(继承自C语言)实现。`new`分配并初始化对象内存,`delete`释放并调用析构函数;而`malloc`等函数仅处理裸内存,不涉及构造与析构。掌握这些可有效管理内存,避免泄漏和悬空指针问题。智能指针如`std::unique_ptr`和`std::shared_ptr`能自动管理内存,确保异常安全。关注威哥爱编程,了解更多全栈开发技巧。 先赞再看后评论,腰缠万贯财进门。
什么是内存泄漏?C++中如何检测和解决?
大家好,我是V哥。内存泄露是编程中的常见问题,可能导致程序崩溃。特别是在金三银四跳槽季,面试官常问此问题。本文将探讨内存泄露的定义、危害、检测方法及解决策略,帮助你掌握这一关键知识点。通过学习如何正确管理内存、使用智能指针和RAII原则,避免内存泄露,提升代码健壮性。同时,了解常见的内存泄露场景,如忘记释放内存、异常处理不当等,确保在面试中不被秒杀。最后,预祝大家新的一年工作顺利,涨薪多多!关注威哥爱编程,一起成为更好的程序员。
【C语言】内存管理函数详细讲解
在C语言编程中,内存管理是至关重要的。动态内存分配函数允许程序在运行时请求和释放内存,这对于处理不确定大小的数据结构至关重要。以下是C语言内存管理函数的详细讲解,包括每个函数的功能、标准格式、示例代码、代码解释及其输出。
130 6
【c++】动态内存管理
本文介绍了C++中动态内存管理的新方式——`new`和`delete`操作符,详细探讨了它们的使用方法及与C语言中`malloc`/`free`的区别。文章首先回顾了C语言中的动态内存管理,接着通过代码实例展示了`new`和`delete`的基本用法,包括对内置类型和自定义类型的动态内存分配与释放。此外,文章还深入解析了`operator new`和`operator delete`的底层实现,以及定位new表达式的应用,最后总结了`malloc`/`free`与`new`/`delete`的主要差异。
89 3
【c++】类和对象(上)(类的定义格式、访问限定符、类域、类的实例化、对象的内存大小、this指针)
本文介绍了C++中的类和对象,包括类的概念、定义格式、访问限定符、类域、对象的创建及内存大小、以及this指针。通过示例代码详细解释了类的定义、成员函数和成员变量的作用,以及如何使用访问限定符控制成员的访问权限。此外,还讨论了对象的内存分配规则和this指针的使用场景,帮助读者深入理解面向对象编程的核心概念。
286 4
在 C++中,realloc 函数返回 NULL 时,需要手动释放原来的内存吗?
在 C++ 中,当 realloc 函数返回 NULL 时,表示内存重新分配失败,但原内存块仍然有效,因此需要手动释放原来的内存,以避免内存泄漏。
【c语言】字符串函数和内存函数
本文介绍了C语言中常用的字符串函数和内存函数,包括`strlen`、`strcpy`、`strcat`、`strcmp`、`strstr`、`strncpy`、`strncat`、`strncmp`、`strtok`、`memcpy`、`memmove`和`memset`等函数的使用方法及模拟实现。文章详细讲解了每个函数的功能、参数、返回值,并提供了具体的代码示例,帮助读者更好地理解和掌握这些函数的应用。
78 0