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



相关文章
|
9天前
|
程序员 C语言
C语言库函数 — 内存函数(含模拟实现内存函数)
C语言库函数 — 内存函数(含模拟实现内存函数)
19 0
|
20天前
|
编译器 C语言 C++
【C语言】memset()函数(内存块初始化函数)
【C语言】memset()函数(内存块初始化函数)
24 0
|
20天前
|
编译器 C语言 C++
【C语言】memcpy()函数(内存块拷贝函数)
【C语言】memcpy()函数(内存块拷贝函数)
38 0
|
8天前
|
编译器 C语言 C++
【C++初阶(九)】C++模版(初阶)----函数模版与类模版
【C++初阶(九)】C++模版(初阶)----函数模版与类模版
16 0
|
18天前
|
存储 缓存 C++
C++链表常用的函数编写(增查删改)内附完整程序
C++链表常用的函数编写(增查删改)内附完整程序
|
20天前
|
存储 安全 编译器
【C++】类的六大默认成员函数及其特性(万字详解)
【C++】类的六大默认成员函数及其特性(万字详解)
35 3
|
20天前
|
编译器 C语言 C++
【C语言】calloc()函数详解(动态内存开辟函数)
【C语言】calloc()函数详解(动态内存开辟函数)
24 0
|
20天前
|
存储 前端开发 编译器
【C语言】memmove()函数(拷贝重叠内存块函数详解)
【C语言】memmove()函数(拷贝重叠内存块函数详解)
32 1
|
23天前
|
存储 Linux C语言
【C++练级之路】【Lv.5】动态内存管理(都2023年了,不会有人还不知道new吧?)
【C++练级之路】【Lv.5】动态内存管理(都2023年了,不会有人还不知道new吧?)
|
23天前
|
安全 程序员 C++
【C++ 基本知识】现代C++内存管理:探究std::make_系列函数的力量
【C++ 基本知识】现代C++内存管理:探究std::make_系列函数的力量
97 0