C++中三种创建对象的方法【转】
我们都知道C++中有三种创建对象的方法,如下:
#include <iostream>
using namespace std; class A
{
private:
int n;
public:
A(int m):n(m)
{
}
~A(){}
}; int main()
{
A a(); //栈中分配
A b = A(); //栈中分配
A* c = new A(); //堆中分配
delete c;
return ;
}
第一种和第二种没什么区别,一个隐式调用,一个显式调用,两者都是在进程虚拟地址空间中的栈中分配内存,而第三种使用了new,在堆中分配了内存,而栈中内存的分配和释放是由系统管理,而堆中内存的分配和释放必须由程序员手动释放,所以这就产生一个问题是把对象放在栈中还是放在堆中的问题,这个问题又和堆和栈本身的区别有关:
这里面有几个问题:
1.堆和栈最大可分配的内存的大小
2.堆和栈的内存管理方式
3.堆和栈的分配效率
首先针对第一个问题,一般来说对于一个进程栈的大小远远小于堆的大小,在linux中,你可以使用ulimit -s (单位kb)来查看一个进程栈的最大可分配大小,一般来说不超过8M,有的甚至不超过2M,不过这个可以设置,而对于堆你会发现,针对一个进程堆的最大可分配的大小在G的数量级上,不同系统可能不一样,比如32位系统最大不超过2G,而64为系统最大不超过4G,所以当你需要一个分配的大小的内存时,请用new,即用堆。
其次针对第二个问题,栈是系统数据结构,对于进程/线程是唯一的,它的分配与释放由操作系统来维护,不需要开发者来管理。在执行函数时,函数内局部变量的存储单元都可以在栈上创建,函数执行结束时,这些存储单元会被自动释放。栈内存分配运算内置于处理器的指令集中,效率很高,不同的操作系统对栈都有一定的限制。 堆上的内存分配,亦称动态内存分配。程序在运行的期间用malloc申请的内存,这部分内存由程序员自己负责管理,其生存期由开发者决定:在何时分配,分配多少,并在何时用free来释放该内存。这是唯一可以由开发者参与管理的内存。使用的好坏直接决定系统的性能和稳定。
由上可知,但我们需要的内存很少,你又能确定你到底需要多少内存时,请用栈。而当你需要在运行时才知道你到底需要多少内存时,请用堆。
最后针对第三个问题,栈是机器系统提供的数据结构,计算机会在底层对栈提供支持:分配专门的寄存器存放栈的地址,压栈出栈都有专门的指令执行,这就决定了栈的效率 比较高。堆则是C/C++函数库提供的,它的机制是很复杂的,例如为了分配一块内存,库函数会按照一定的算法(具体的算法可以参考数据结构/操作系统)在 堆内存中搜索可用的足够大小的空间,如果没有足够大小的空间(可能是由于内存碎片太多),就有可能调用系统功能去增加程序数据段的内存空间,这样就有机会 分 到足够大小的内存,然后进行返回。显然,堆的效率比栈要低得多。
再补充几点。
平时使用容器的时候,都会直接使用上述第一种方法定义。
比如:
std::vector<TestBase> a;
那么如果a要存很大的东西,会不会爆栈呢。
其实是不会的,虽然a是在栈上分配的内存,但是a所占的栈空间是很小的。
std::cout << sizeof(a) << std::endl;
输出: 24
比如vector,无论push了多少东西,存的类有多大,都在在堆上new的,而在栈上所占用的空间就只有三个指针的大小。
当然你也可以new 一个容器。这样占用栈的空间就只有一个指针的大小了。
最新文章
- make: *** [out/host/linux-x86/obj/EXECUTABLES/aidl_intermediates/aidl] 错误 1,make: *** [out/host/linux-x86/obj/lib/libESR_Portable.so] 错误 1
- linux启动时文件系统错误问题
- c语言操作一维数组-3
- HDU 4417 Super Mario(划分树+二分)
- java中hashCode()方法的作用
- MVC3+AutoFac实现程序集级别的依赖注入
- hadoop小文件合并
- 计数dp-hdu-4054-Number String
- 使用dropload.js插件进行下拉刷新
- spring boot 如何添加拦截
- 移动web-bootstrap
- post数据过多时,有可能会丢失数据
- hdu1080 DP(类最长公共子序列)
- 刚刚明白了for循环写三角形
- 基于python的接口测试框架设计(三)接口测试的框架
- Treasure Exploration---poj2594(传递闭包Floyd+最小路径覆盖)
- PHP 对字符串 进行填充
- java实现FTP下载文件
- vue.js 源代码学习笔记 ----- 工具方法 debug
- index、noindex、follow、nofollow的使用说明
热门文章
- (转)并行编译 Xoreax IncrediBuild
- Git -- 工作区 和 暂存区
- 微服务架构eureka集群高可用配置
- Android 安全提示 笔记
- 【转】Java中的多线程学习大总结
- mysql中使用show variables同时查询多个参数值?show variables的使用?
- [Converge] Backpropagation Algorithm
- openssl &; openssh
- iOS 播放gif动态图的方式探讨
- 【RF库Collections测试】lists should be equal