C++内存分配方式——小结
1 内存分配方式
内存分配方式有如下三种:
- 从静态存储区域分配。内存在程序编译的时候就分配好了,这些内存在整个程序运行期间都存在,如全局变量、static变量等等。
- 在堆栈上分配。在函数执行期间,函数形参、函数内局部变量的存储单元都置于堆栈上,函数调用结束后自动从堆栈上释放。
- 从堆(heap)或自由存储空间分配,也叫动态内存分配。程序运行期间使用malloc()或new申请内存,使用free()或delete释放内存。
一般的原则是:如果使用堆栈和静态存储就能够满足应用需求,就不要使用动态存储。因为,在堆上动态分配内存需要很多额外开销。
2 常见的内存错误及其对策
- 内存分配未成功就使用了它——在使用内存之前检查指针是否为NULL。
- 内存分配虽然成功,但是尚未初始化就使用了它——建立初始化意识。
- 内存分配成功并且初始化,但是操作越过了内存的边界。
- 忘记释放内存或只释放了部分内存,造成内存泄漏。另外动态内存的申请和释放必须匹配。
- 释放了内存还在继续使用它。
- 使用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++内存分配方式——小结的更多相关文章
- c/c++内存分配方式(转)
原文链接:http://blog.csdn.net/jing0611/article/details/4030237 1.内存分配方式 内存分配方式有三种: [1]从静态存储区域分配.内存在 程序编译 ...
- 内存分配方式,堆区,栈区,new/delete/malloc/free
1.内存分配方式 内存分配方式有三种: [1]从静态存储区域分配.内存在程序编译的时候就已经分配好,这块内存在程序的整个运行期间都存在.例如全局变量,static变量. [2]在栈上创建.在执行函数时 ...
- [转载]C语言程序的内存分配方式
"声明一个数组时,编译器将根据声明所指定的元素数量为数量为数组保留内存空间."其实就是编译器在编译的过程中,会加入几条汇编指令在程序里处理内存分配,并不是说编译时就分配了内存,不要 ...
- 转 C/C++内存分配方式与存储区
C/C++内存分配方式与存储区 C/C++内存分配有三种方式:[1]从静态存储区域分配.内存在程序编译的时候就已经分配好,这块内存在程序的整个运行期间都存在.例如全局变量,static变量.[2]在栈 ...
- C和C++内存分配方式记录
C. C++中内存分配方式可以分为三种: (1)从静态存储区域分配:内存在程序编译时就已经分配好,这块内存在程序的整个运行期间都存在.速度快.不容易出错,因为有系统会善后.例如全局变量,static变 ...
- C/C++ 内存分配方式,堆区,栈区,new/delete/malloc/free
内存分配方式 内存分配方式有三种: [1] 从静态存储区域分配.内存在程序编译的时候就已经分配好,这块内存在程序的整个运行期间都存在.例如全局变量, static 变量. [2] 在栈上创建.在执行函 ...
- C++内存分配方式——(别人的博客)
http://www.cnblogs.com/easonpan/archive/2012/04/26/2471153.html http://blog.csdn.net/chen825919148/a ...
- C++内存分配方式详解——堆、栈、自由存储区、全局/静态存储区和常量存储区
栈,就是那些由编译器在需要的时候分配,在不需要的时候自动清除的变量的存储区.里面的变量通常是局部变量.函数参数等.在一个进程中,位于用户虚拟地址空间顶部的是用户栈,编译器用它来实现函数的调用.和堆一样 ...
- C中内存分配方式[转载]
在C 中,内存分成5个区,他们分别是堆.栈.自由存储区.全局/静态存储区和常量存储区. 一.简介: 1.栈,就是那些由编译器在需要的时候分配,在无需的时候自动清除的变量的存储区.里面的变量通常是局部变 ...
随机推荐
- Linux终端 Tab 补全命令
1. vi编辑器打开 /etc/bash.bashrc文件 vi /etc/bash.bashrc 2.找到文件中的下列代码 3.将注释符号#去掉,即改成 4.最后 source一下 /etc/bas ...
- jsoup解析页面
package com.java.jsoup; /** * jsoup解析网页 * @author nidegui * @version 2019年4月29日 下午5:12:02 * */ impor ...
- Python基础:dict & set
一 :dict 1:Python内置了字典:dict的支持,dict全称dictionary,在其他语言中也称为map,使用键-值(key-value)存储,具有极快的查找速度. eg: dict查找 ...
- 【JavaScript框架封装】实现一个类似于JQuery的缓存框架的封装
// 缓存框架 (function (xframe) { /** * 实现了缓存框架的临时存储功能(内存存储) * @type {{data: Array, get: (function(*): *) ...
- [luogu 1270] “访问”美术馆 (树形dp)
传送门 Description 经过数月的精心准备,Peer Brelstet,一个出了名的盗画者,准备开始他的下一个行动.艺术馆的结构,每条走廊要么分叉为两条走廊,要么通向一个展览室.Peer知道每 ...
- 【codeforces 731D】80-th Level Archeology
[题目链接]:http://codeforces.com/contest/731/problem/D [题意] 给你n个象形文; 每个象形文由l[i]个数字组成; 你可以把所有的组成象形文的数字同时增 ...
- C#--文件操作的一些技巧
Using的特点 Using 打开什么,就自动关闭什么,using中包含的其他类是否关闭,using是不管的 XML文档读取 重点:必须是标准的xml文档,否则会出错 string xmlxx = @ ...
- POJ 2018
又一水,设dp[i]为以i结尾的有最大平均值的起始位置. #include <iostream> #include <cstdio> #include <cstring& ...
- Java数据结构(排序篇)
冒泡排序:是经过n-1趟子排序完毕的,第i趟子排序从第1个数至第n-i个数,若第i个数比后一个数大(则升序,小则降序)则交换两数.大泡在上,小泡在下. 选择排序:每一趟从待排序的数据元素中选出最小(或 ...
- 用友ERP T6技术解析(六) 库龄分析
2.4 库存管理 2.4.1 库龄分析 介绍:库存账龄是在某时间节点,某种或某类存货的库存时间的加权平均值,跟库存周转率关系明显,库存周转率越高,库存账龄越低,可是二者又不是反比关系.不能简单把库 ...