1 内存分配方式

内存分配方式有如下三种:

  1. 从静态存储区域分配。内存在程序编译的时候就分配好了,这些内存在整个程序运行期间都存在,如全局变量、static变量等等。
  2. 在堆栈上分配。在函数执行期间,函数形参、函数内局部变量的存储单元都置于堆栈上,函数调用结束后自动从堆栈上释放。
  3. 从堆(heap)或自由存储空间分配,也叫动态内存分配。程序运行期间使用malloc()或new申请内存,使用free()或delete释放内存。

一般的原则是:如果使用堆栈和静态存储就能够满足应用需求,就不要使用动态存储。因为,在堆上动态分配内存需要很多额外开销。

2 常见的内存错误及其对策

  1. 内存分配未成功就使用了它——在使用内存之前检查指针是否为NULL。
  2. 内存分配虽然成功,但是尚未初始化就使用了它——建立初始化意识。
  3. 内存分配成功并且初始化,但是操作越过了内存的边界。
  4. 忘记释放内存或只释放了部分内存,造成内存泄漏。另外动态内存的申请和释放必须匹配。
  5. 释放了内存还在继续使用它。
    • 使用free()或者delete释放了内存之后,没有把指针设置为NULL,产生“野指针”。
    • 函数return返回指向堆栈内存的指针或引用,因为该函数结束时会自动销毁。
    • 多次释放同一块内存。

3 指针参数传递内存

如果函数的参数是一个指针,不要使用它去申请一块内存。因为编译器总是为函数的实参制作临时副本,指针p的副本_p值一样,因此指向同一块内存。但是如果使用_p来申请内存,p指向的内存则完全没有改变,反而在函数结束后造成内存的泄漏。

4 free和delete与指针

free()和delete只是把指针所指向的内存释放掉,并没有把指针本身删掉,大多数情况下p指向的地址仍不变,变成了“野指针”。

  • 指针消亡了不代表指针指向的内容消亡了。
  • 内存被释放了不代表指针消亡了或变成NULL。

“野指针”有以下的成因

  • 没有初始化指针变量。
  • 指针被free()或者delete之后,没有置为NULL。
  • 指针超越了变量的作用范围

5 malloc/free和new/delete

malloc()与free()是C++/C语言的标准库函数,new/delete是C++的运算符,它们都可以动态申请和释放内存。

对于非内部数据类型(如ADT/UDT)的对象而言,使用malloc()与free()无法满足要求:对象创建自动调用构造函数,对象销毁自动调用析构函数。由于malloc()与free()是库函数不是运算符,不在编译器控制权限之内,因此C++语言使用new/delete。

6 malloc/free使用要点

函数malloc原型如下:

void *malloc(size_t size);

例如申请一块长度为n的整型数组的内存:

int *p = (int *)malloc(sizeof(int) * n);

  • malloc()函数返回值的类型是void *,所以在调用malloc要显式地进行类型转换。
  • malloc()只关心内存的总字数,不知道要申请的内存是什么类型,使用sizeof运算符计算类型的大小。

函数free的原型如下:

void free(void * memblock);

7 new的3种使用方式

new有3种使用方式:plain new、nothrow new、placement new。

  • plain new:普通的new,在失败后抛出标准异常std::bad_alloc——形式:new
  • nothrow new:在失败后不抛出异常,而像过去一样检查返回值是否为NULL——形式:new(nothrow)
  • placement new:允许在一块已经分配成功的内存上重新构造对象或者数组——形式:new(p)

8 new/delete使用要点

new内置了sizeof、类型转换和类型安全检查功能,比使用malloc简单很多,例如。

int *p = (int *)malloc(sizeof(int) * n);
int *p = new int[n]; Obj *o = new Obj; //使用默认构造函数构造对象
Obj *o = new Obj(); //使用初值1构造对象
Obj *o = new Obj[]; //如用new创建对象数组,只能使用默认构造函数 delete p;  //释放一个对象
delete []o;  //释放一个对象数组
  • 无论何种类型,new/delete和new[]/delete[]必须正确搭配使用
  • 多次delete一个不等于NULL的指针会导致运行时错误。

(完)

C++内存分配方式——小结的更多相关文章

  1. c/c++内存分配方式(转)

    原文链接:http://blog.csdn.net/jing0611/article/details/4030237 1.内存分配方式 内存分配方式有三种: [1]从静态存储区域分配.内存在 程序编译 ...

  2. 内存分配方式,堆区,栈区,new/delete/malloc/free

    1.内存分配方式 内存分配方式有三种: [1]从静态存储区域分配.内存在程序编译的时候就已经分配好,这块内存在程序的整个运行期间都存在.例如全局变量,static变量. [2]在栈上创建.在执行函数时 ...

  3. [转载]C语言程序的内存分配方式

    "声明一个数组时,编译器将根据声明所指定的元素数量为数量为数组保留内存空间."其实就是编译器在编译的过程中,会加入几条汇编指令在程序里处理内存分配,并不是说编译时就分配了内存,不要 ...

  4. 转 C/C++内存分配方式与存储区

    C/C++内存分配方式与存储区 C/C++内存分配有三种方式:[1]从静态存储区域分配.内存在程序编译的时候就已经分配好,这块内存在程序的整个运行期间都存在.例如全局变量,static变量.[2]在栈 ...

  5. C和C++内存分配方式记录

    C. C++中内存分配方式可以分为三种: (1)从静态存储区域分配:内存在程序编译时就已经分配好,这块内存在程序的整个运行期间都存在.速度快.不容易出错,因为有系统会善后.例如全局变量,static变 ...

  6. C/C++ 内存分配方式,堆区,栈区,new/delete/malloc/free

    内存分配方式 内存分配方式有三种: [1] 从静态存储区域分配.内存在程序编译的时候就已经分配好,这块内存在程序的整个运行期间都存在.例如全局变量, static 变量. [2] 在栈上创建.在执行函 ...

  7. C++内存分配方式——(别人的博客)

    http://www.cnblogs.com/easonpan/archive/2012/04/26/2471153.html http://blog.csdn.net/chen825919148/a ...

  8. C++内存分配方式详解——堆、栈、自由存储区、全局/静态存储区和常量存储区

    栈,就是那些由编译器在需要的时候分配,在不需要的时候自动清除的变量的存储区.里面的变量通常是局部变量.函数参数等.在一个进程中,位于用户虚拟地址空间顶部的是用户栈,编译器用它来实现函数的调用.和堆一样 ...

  9. C中内存分配方式[转载]

    在C 中,内存分成5个区,他们分别是堆.栈.自由存储区.全局/静态存储区和常量存储区. 一.简介: 1.栈,就是那些由编译器在需要的时候分配,在无需的时候自动清除的变量的存储区.里面的变量通常是局部变 ...

随机推荐

  1. Linux终端 Tab 补全命令

    1. vi编辑器打开 /etc/bash.bashrc文件 vi /etc/bash.bashrc 2.找到文件中的下列代码 3.将注释符号#去掉,即改成 4.最后 source一下 /etc/bas ...

  2. jsoup解析页面

    package com.java.jsoup; /** * jsoup解析网页 * @author nidegui * @version 2019年4月29日 下午5:12:02 * */ impor ...

  3. Python基础:dict & set

    一 :dict 1:Python内置了字典:dict的支持,dict全称dictionary,在其他语言中也称为map,使用键-值(key-value)存储,具有极快的查找速度. eg: dict查找 ...

  4. 【JavaScript框架封装】实现一个类似于JQuery的缓存框架的封装

    // 缓存框架 (function (xframe) { /** * 实现了缓存框架的临时存储功能(内存存储) * @type {{data: Array, get: (function(*): *) ...

  5. [luogu 1270] “访问”美术馆 (树形dp)

    传送门 Description 经过数月的精心准备,Peer Brelstet,一个出了名的盗画者,准备开始他的下一个行动.艺术馆的结构,每条走廊要么分叉为两条走廊,要么通向一个展览室.Peer知道每 ...

  6. 【codeforces 731D】80-th Level Archeology

    [题目链接]:http://codeforces.com/contest/731/problem/D [题意] 给你n个象形文; 每个象形文由l[i]个数字组成; 你可以把所有的组成象形文的数字同时增 ...

  7. C#--文件操作的一些技巧

    Using的特点 Using 打开什么,就自动关闭什么,using中包含的其他类是否关闭,using是不管的 XML文档读取 重点:必须是标准的xml文档,否则会出错 string xmlxx = @ ...

  8. POJ 2018

    又一水,设dp[i]为以i结尾的有最大平均值的起始位置. #include <iostream> #include <cstdio> #include <cstring& ...

  9. Java数据结构(排序篇)

    冒泡排序:是经过n-1趟子排序完毕的,第i趟子排序从第1个数至第n-i个数,若第i个数比后一个数大(则升序,小则降序)则交换两数.大泡在上,小泡在下. 选择排序:每一趟从待排序的数据元素中选出最小(或 ...

  10. 用友ERP T6技术解析(六) 库龄分析

    2.4 库存管理   2.4.1 库龄分析 介绍:库存账龄是在某时间节点,某种或某类存货的库存时间的加权平均值,跟库存周转率关系明显,库存周转率越高,库存账龄越低,可是二者又不是反比关系.不能简单把库 ...