内存泄漏&栈溢出

  C++中,我们主要涉及的内存是栈和堆,

堆  (By programmer) 申请后由程序员主动释放,遗忘后果严重;

栈 (By compiler)需要时由编译器分配,在不需要时自动清除的存储区。一般用于存放局部变量、函数参数。

这些存放在栈中的数据只在当前函数及下一层函数中有效,一旦函数返回了,会发生自动释放。

除此外还有

静态存储区 全局&静态变量在同一块内存中;

常量存储区 常量、不允许修改;

代码区 不允许修改、可执行;

回到发生内存泄漏的场景。

  通常来说,一个线程的栈内存是有限的。栈上的内存通常是由编译器来自动管理的。当在栈上分配一个新的变量时,或进入一个函数时,栈的指针会下移,相当于在栈上分配了一块内存,当这个变量的生命周期结束时,栈的指针会上移。

  由于栈上的内存的分配和回收都是由编译器控制的,所以在栈上是不会发生内存泄露的,只会发生栈溢出(Stack Overflow),也就是分配的空间超过了规定的栈大小。而当在堆上申请内存后忘记使用 free/delete 来回收,就发生了内存泄露。

内存泄漏检测

https://www.cnblogs.com/skynet/archive/2011/02/20/1959162.html   不赘述

野指针

  指向一个已删除的对象或未申请访问受限内存区域的指针。

  需要通过养成良好的编程习惯来尽力减少。对野指针进行操作很容易造成程序错误。

  • 指针变量未初始化

  创建新指针时应该对其初始化,设置为NULL或者指向合法的内存,不然会被随机指向任意空间。

  • 野指针指针释放后之后未置空

  free或delete只是把指针所指的内存给释放掉,但并没有结束指针本身。释放后的指针应立即将指针置为NULL,防止产生“野指针”。

  • 野指针指针操作超越变量作用域

不要返回指向栈内存的指针或引用,因为栈内存在函数结束时会被释放。

class A
{
public:
void Func(void)
{
cout << “Func of class A” << endl;
}
}; class B
{
public:
A *p;
void Test(void)
{
A a;
p = &a; //a是函数Test中的局部量,而不属于整个class B
}
void Test1()
{
p->Func(); // p 是“野指针”
}
};

  a在void Test(void)函数结束时会被释放,所以在函数void Test1()中的指针p就是野指针。

在堆上,还是在栈上?

如果需要在堆上创建对象,则使用new运算符或者malloc系列函数。

Object obj;

  有什么含义?

使对象obj具有“自动存储(automatic storage)”的性质。所谓“自动存储”,意思是这个对象的存储位置取决于其声明所在的上下文。

  如果这个语句出现在函数内部,那么它就在栈上创建对象。

  如果这个语句不是在函数内部,而是作为一个类的成员变量,则取决于这个类的对象发生分配时的实际场景。

1.

class Class
{
Object obj;
}; Class *pC = new Class;

  类的指针pC在堆上创建,于是pC->obj也在堆上创建。

2.

Object *pObj;
pObj = new Object;

  同意,指针pObj是“自动存储”的。

  而下面一行语句则指出,这个指针所指向的对象是在堆上面分配的。如果这两行语句出现在一个函数内部,意味着当函数结束时,pObj会被销毁,但是它指向的对象不会。因此,为了继续使用这个对象,通常我们会在函数最后添加一个return语句,或者使用一个传出参数。否则的话,这个在堆上创建的对象就没有指针指向它,换而言之,发生了内存泄漏。

3.指针指向一个在栈上创建的对象:

Object obj;
Object *pObj = &obj;

  

4.有些情况,即便你在栈上创建了对象,它还是会占用堆的空间

void func
{
std::vector v;
}

  对象v在栈上,因为vector类在堆上存储数据,即v所管理的数据在堆上。

堆和栈

  堆和栈的区别在于两点:

  1. 生命周期
  2. 性能

  第一点才是我们需要着重考虑的。在栈上的对象不需要我们手动管理内存,所以推荐能在栈上创建变量就老实一点。然而如果一个必要的比其所在的上下文更长生命周期的对象变量需要被创建的时候,需要慎重的去考虑。

  除了函数内部的变量和成员变量。还有两类变量:全局变量和static变量:它们即不在堆上创建,也不在栈上创建。它们有自己的内存空间,是除堆和栈以外的数据区。也就是说,当实例对象即不在函数内部,又不是类的成员变量时,这个对象会在全局数据段创建,同理适用于static变量。

  对于指针,如果这个语句出现在函数内部或类的成员变量,正如我们前面所说的,这个指针是自动存储的。但是,如果这个语句是在类的外部,它就是在全局数据段创建的。虽然它指向的对象可能在堆上创建,也可能在栈上创建。

堆和栈的内存分布&一些关于内存泄露、栈溢出和野指针的内容(头秃的更多相关文章

  1. JVM内存结构之堆、栈、方法区以及直接内存、堆和栈区别

    JVM内存结构之堆.栈.方法区以及直接内存.堆和栈区别 一.  理解JVM中堆与栈以及方法区 堆(heap):FIFO(队列优先,先进先出):二级缓存:*JVM中只有一个堆区被所有线程所共享:对象和数 ...

  2. .NET的堆和栈03,引用类型对象拷贝以及内存分配

    在" .NET的堆和栈01,基本概念.值类型内存分配"中,了解了"堆"和"栈"的基本概念,以及值类型的内存分配.我们知道:当执行一个方法的时 ...

  3. binary hacks读数笔记(堆、栈 VMA的分布)

    一.首先看一个简单的程序: #include<stdlib.h> int main() { while(1) { sleep(1000); } return 0; } gcc -stati ...

  4. .NET的堆和栈01,基本概念、值类型内存分配

    当我们对.NET Framework的一些基本面了解之后,实际上,还是很有必要了解一些更底层的知识.比如.NET Framework是如何进行内存管理的,是如何垃圾回收的......这样,我们才能写出 ...

  5. (十一)C语言中内存堆和栈的区别

    在计算机领域,堆栈是一个不容忽视的概念,我们编写的C语言程序基本上都要用到.但对于很多的初学着来说,堆栈是一个很模糊的概念. 堆栈:一种数据结构.一个在程序运行时用于存放的地方,这可能是很多初学者的认 ...

  6. NET的堆和栈04,对托管和非托管资源的垃圾回收以及内存分配

    在" .NET的堆和栈01,基本概念.值类型内存分配"中,了解了"堆"和"栈"的基本概念,以及值类型的内存分配.我们知道:当执行一个方法的时 ...

  7. .NET的堆和栈02,值类型和引用类型参数传递以及内存分配

    在" .NET的堆和栈01,基本概念.值类型内存分配"中,了解了"堆"和"栈"的基本概念,以及值类型的内存分配.我们知道:当执行一个方法的时 ...

  8. 栈 堆 stack heap 堆内存 栈内存 内存分配中的堆和栈 掌握堆内存的权柄就是返回的指针 栈是面向线程的而堆是面向进程的。 new/delete and malloc/ free 指针与内存模型

    小结: 1.栈内存 为什么快? Due to this nature, the process of storing and retrieving data from the stack is ver ...

  9. Java堆和栈详解

    Java把内存分成两种,一种叫做栈内存,一种叫做堆内存 在函数中定义的一些基本类型的变量和对象的引用变量都是在函数的栈内存中分配.当在一段代码块中定义一个变量时,java就在栈中为这个变量分配内存空间 ...

随机推荐

  1. 基于Hadoop集群搭建Hive安装与配置(yum插件安装MySQL)---linux系统《小白篇》

    用到的安装包有: apache-hive-1.2.1-bin.tar.gz mysql-connector-java-5.1.49.tar.gz 百度网盘链接: 链接:https://pan.baid ...

  2. [BD] 阿里云部署hadoop集群

    安装方式 rpm包安装:下载rpm文件后离线装,安装过程中会下载相应依赖 bin文件安装:在线安装 tar包安装 步骤 下载安装文件:买香港机器,按量付费,传到windows电脑 购买三台,按需付费, ...

  3. 联想ThinkServer服务器安装CentOS7 Redhat7系统 驱动R110i RAID卡

    1.下载对应版本的驱动(因为联想没有CentOS的驱动用redhat的驱动就可以). 2.进入BIOS里,在高级设置里找到SATA设置,把SATA模式改成RAID(重启后配置raid),sSATA模式 ...

  4. localectl set-locale LANG=en_US.UTF-8

    localectl set-locale LANG=en_US.UTF-8    

  5. /etc/ssh/sshd_config ssh自动断 cent7

    vim /etc/ssh/sshd_config ClientAliveInterval 60ClientAliveCountMax 8630000 ClientAliveInterval 30Cli ...

  6. python基础之常用模块一(sys、greenlet、pymysql、paramiko、pexpect、configparser)

    一.sys模块(内置模块) 用于提供对解释器相关的操作 import syssys.argv 命令行参数List,第一个元素是程序本身路径 sys.exit(n) 退出程序,正常退出时exit(0) ...

  7. Linux中级之netfilter/iptables应用及补充

    一.iptables介绍 Netfilter/Iptables(以下简称Iptables)是unix/linux自带的一款优秀且开放源代码的完全自由的基于包过滤的防火墙工具,它的功能十分强大,使用非常 ...

  8. MyBatis 高级查询之一对一查询(九)

    高级查询之一对一查询 查询条件:根据游戏角色ID,查询账号信息 我们在之前创建的映射器接口 GameMapper.java 中添加接口方法,如下: /** * 根据角色ID查询账号信息 * @para ...

  9. #undef 与 exit(0) 使用

    #undef  与 #defined 反,实际使用中较多的是当你需要使用自己定义的标准C里面已经的函数时可以这样操作: exit(0)和exit(1)是系统判断函数是否有正常的退出,一般0表示正常的退 ...

  10. Spring Mvc Long类型精度丢失

    背景 在使用Spring Boot Mvc的项目中,使用Long类型作为id的类型,但是当前端使用Number类型接收Long类型数据时,由于前端精度问题,会导致Long类型数据转换为Number类型 ...