C++用new创建对象和不用new创建对象的区别解析_C 语言

我们都知道C++中有三种创建对象的方法,如下:

复制代码 代码如下:

#include <iostream>
using namespace std;

class A
{
private:
    int n;
public:
    A(int m):n(m)
    {
    }
    ~A(){}
};

int main()
{
    A a(1);  //栈中分配
    A b = A(1);  //栈中分配
    A* c = new A(1);  //堆中分配
  delete c;
    return 0;
}

第一种和第二种没什么区别,一个隐式调用,一个显式调用,两者都是在进程虚拟地址空间中的栈中分配内存,而第三种使用了new,在堆中分配了内存,而栈中内存的分配和释放是由系统管理,而堆中内存的分配和释放必须由程序员手动释放,所以这就产生一个问题是把对象放在栈中还是放在堆中的问题,这个问题又和堆和栈本身的区别有关:

这里面有几个问题:
1.堆和栈最大可分配的内存的大小
2.堆和栈的内存管理方式
3.堆和栈的分配效率

首先针对第一个问题,一般来说对于一个进程栈的大小远远小于堆的大小,在linux中,你可以使用ulimit -s (单位kb)来查看一个进程栈的最大可分配大小,一般来说不超过8M,有的甚至不超过2M,不过这个可以设置,而对于堆你会发现,针对一个进程堆的最大可分配的大小在G的数量级上,不同系统可能不一样,比如32位系统最大不超过2G,而64为系统最大不超过4G,所以当你需要一个分配的大小的内存时,请用new,即用堆。

其次针对第二个问题,栈是系统数据结构,对于进程/线程是唯一的,它的分配与释放由操作系统来维护,不需要开发者来管理。在执行函数时,函数内局部变量的存储单元都可以在栈上创建,函数执行结束时,这些存储单元会被自动释放。栈内存分配运算内置于处理器的指令集中,效率很高,不同的操作系统对栈都有一定的限制。 堆上的内存分配,亦称动态内存分配。程序在运行的期间用malloc申请的内存,这部分内存由程序员自己负责管理,其生存期由开发者决定:在何时分配,分配多少,并在何时用free来释放该内存。这是唯一可以由开发者参与管理的内存。使用的好坏直接决定系统的性能和稳定。

由上可知,但我们需要的内存很少,你又能确定你到底需要多少内存时,请用栈。而当你需要在运行时才知道你到底需要多少内存时,请用堆。

最后针对第三个问题,栈是机器系统提供的数据结构,计算机会在底层对栈提供支持:分配专门的寄存器存放栈的地址,压栈出栈都有专门的指令执行,这就决定了栈的效率 比较高。堆则是C/C++函数库提供的,它的机制是很复杂的,例如为了分配一块内存,库函数会按照一定的算法(具体的算法可以参考数据结构/操作系统)在 堆内存中搜索可用的足够大小的空间,如果没有足够大小的空间(可能是由于内存碎片太多),就有可能调用系统功能去增加程序数据段的内存空间,这样就有机会 分 到足够大小的内存,然后进行返回。显然,堆的效率比栈要低得多。

由上可知,能用栈则用栈。

复制代码 代码如下:

#include <stdio.h>
#include <stdlib.h> 
void main()
{
 int n,*p,i,j,m;
 printf("本程序可对任意个整数排序;\n");
 printf("请输入整数的总个数: ");
 scanf("%d",&n);
 p=(int *)calloc(n,sizeof(int));    //运行时决定内存分配大小
 if(p==0)   {
  printf("分配失败!\n"); 
  exit(1); 
 }

时间: 2024-09-13 20:08:56

C++用new创建对象和不用new创建对象的区别解析_C 语言的相关文章

C++用new创建对象和不用new创建对象的区别解析

在C++用new创建对象和不用new创建对象是有区别的,不知你是否清楚的了解它们到底有什么样的区别呢?下面小编就用示例来告诉大家吧,需要的朋友可以过来参考下   我们都知道C++中有三种创建对象的方法,如下: 复制代码 代码如下: #include <iostream> using namespace std; class A { private:     int n; public:     A(int m):n(m)     {     }     ~A(){} }; int main()

【技术贴】解决打开程序出错,提示错误429,activex部件不能创建对象,不用重装系统。

[技术贴]解决打开程序出错,提示错误429,activex部件不能创建对象,不用重装系统. 错误429修复包下载,activex部件不能创建修复包下载 1.最完美解决方案 去这里http://ishare.iask.sina.com.cn/f/17276925.html 下载 错误429修复包 ,双击运行之后即可.ok  (90%可以解决问题!!) 2.去这里下载429处理包 http://ishare.iask.sina.com.cn/f/6258417.html?retcode=0  3.别

流程图-Activiti用Exclusive Gateway和不用Exclusive Gateway的区别

问题描述 Activiti用Exclusive Gateway和不用Exclusive Gateway的区别 这是一组没有用Exclusive Gateway的流程图没有用Exclusive Gateway的发布之后会出现红色框里面的东西.不知道是什么也没研究明白请大神指导指导..小弟在此感激不尽..如果用了Exclusive Gateway这个就不会出现红色框里的东西..

JavaScript中使用var 关键字与不用var关键字的区别

var 关键字作用 声明作用:如声明个变量.  语法 var c = 1;  省略var 在javascript中,若省略var关键字而直接赋值,那么这个变量为全局变量,哪怕是在function里定义的.   <script type="text/javascript">   function Define() {     a = 2;   }   function Hello() {     alert(a);   } </script> 如代码所示,运行函数

c++中堆栈及创建对象示例代码_C 语言

简介 栈(stack),先进后出,位于一级缓存中,操作系统自动分配释放 ,存放函数的参数值,局部变量的值等,被调用时处于存储空间中,调用完毕立即释放. 堆(heap),堆包含一个链表来维护已用和空闲的不连续的内存块,存放在二级缓存中,一般由程序员分配释放. 快速记忆方式: 一级缓存比二级缓存快,栈是一个先进后出列表,存取非常快,所以栈是在一级缓存中. 栈中不能随机取数据,只能取最上面的一个,存放的内容必然要有严格的存取顺序,所以适合函数调用时的形参.局部变量. 栈空间有限,一般PC一级缓存就几M

解决不用sizeof求出int大小的方法_C 语言

代码如下所示: 复制代码 代码如下: #include <stdio.h>  int main(int argc, char *argv[])  {      int a[2];      unsigned int add1 = &a[0];      unsigned int add2 = &a[1];      printf("The address of a[0] is %u/n",add1);      printf("The addres

C语言 解决不用+、-、&amp;#215;、&amp;#247;数字运算符做加法的实现方法_C 语言

题目:写一个函数,求两个整数的之和,要求在函数体内不得使用+.-.×.÷. 分析:这又是一道考察发散思维的很有意思的题目.当我们习以为常的东西被限制使用的时候,如何突破常规去思考,就是解决这个问题的关键所在. 看到的这个题目,我的第一反应是傻眼了,四则运算都不能用,那还能用什么啊?可是问题总是要解决的,只能打开思路去思考各种可能性.首先我们可以分析人们是如何做十进制的加法的,比如是如何得出5+17=22这个结果的.实际上,我们可以分成三步的:第一步只做各位相加不进位,此时相加的结果是12(个位数

求教 :不用OFFICE组件,实现解析、显示、保存Word文档

问题描述 有一项目,要求在不使用OFFICE组件的情况下,来用C#实现解析.显示.保存Word文档.请问各位能否实现.或者在哪里能找到C#这方面的开源代码呢?谢谢- 解决方案 解决方案二:这个项目的需求比较有难度.解决方案三:http://wenku.baidu.com/view/22e2bb10cc7931b765ce1599.html

JavaScript学习笔记之创建对象_javascript技巧

JavaScript 有Date.Array.String等这样的内置对象,功能强大使用简单,人见人爱,但在处理一些复杂的逻辑的时候,内置对象就很无力了,往往需要开发者自定义对象. 从JavaScript定义上讲对象是无序属性的集合,其属性可以包含基本值.对象或函数.也就是说对象是一组没有特定顺序的属性,每个属性会映射到一个值上,是一组键值对,值可以是数据或对象. 对象是JavaScript的基本数据类型.在JavaScript中除了字符串.数字.true.false.null和undefine