开发者社区> 重庆八怪> 正文
阿里云
为了无法计算的价值
打开APP
阿里云APP内打开

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



版权声明:本文内容由阿里云实名注册用户自发贡献,版权归原作者所有,阿里云开发者社区不拥有其著作权,亦不承担相应法律责任。具体规则请查看《阿里云开发者社区用户服务协议》和《阿里云开发者社区知识产权保护指引》。如果您发现本社区中有涉嫌抄袭的内容,填写侵权投诉表单进行举报,一经查实,本社区将立刻删除涉嫌侵权内容。

相关文章
.Net Core实战之基于角色的访问控制的设计(三)
.Net Core实战之基于角色的访问控制的设计(三)
26 0
关于netstat
netstat 命令是 linux 下一个非常有用的命令,顾名思义,其作用是查询当前的网络状态。 以最常用的查询 TCP 连接状态为例 netstat -an|awk '/tcp/ {print $6}'|sort|uniq -c 总结各状态大致意义如下: CLOSED: 初始状态。
820 0
让asp.net默认的上传组件支持进度条反映
对于web下的上传,实际上更多的时候不用上传太大东西,asp.net默认的上传组件足够用了,美中不足就是没有上传进度反映,所以现在要做的就是在asp.net默认的上传基础上加上进度反映。 关于web上传的原理,曾在以前有深入分析过《asp无组件上传进度条解决方案》《Asp无组件上传带进度条(续) 》,并有写过asp版的无组件上传进度条,在这里就不多赘述。
896 0
ASP.NET中如何访问用户控件里面的控件的值
2 楼Eddie005() №.零零伍 ()回复于 2006-08-02 17:19:35 得分 10 最好是在控件里增加一个public的属性访问     public   string   LabelText     {         get{return   this.
683 0
一起谈.NET技术,关于c#静态方法和实例方法的辨析和应用
  本文将围绕c#静态方法和实例方法讨论一下。针对一些观点,如:"静态方法是常驻内存", 还有"静态方法比实例方法先装载",做一个辨析。同时讨论下何时用静态方法,何时用实例方法。   前几日,在微软的好友发给我一个链接:《静态方法和实例化方法之间的区别你知道了嘛? 欢迎讨论!! - 问题最终 ...》,然后说这里某些观点需要澄清一下,希望我写一篇blog。
926 0
一起谈.NET技术,.NET 4的新特性:图表、SEO及可扩展的输出缓存
即将发布的.NET Framework 4.0拥有众多的新改进,此前InfoQ已经对其进行了详尽的报道。本文将再来揭示.NET 4.0中的3个新特性:图表控件、SEO支持以及ASP.NET 4可扩展的输出缓存。
1160 0
+关注
重庆八怪
10年ORACLE/MYSQL DBA,有一定C/C++基础
661
文章
0
问答
文章排行榜
最热
最新
相关电子书
更多
低代码开发师(初级)实战教程
立即下载
阿里巴巴DevOps 最佳实践手册
立即下载
冬季实战营第三期:MySQL数据库进阶实战
立即下载