java怎么new一个数组组,占用内存由多少决定

47143人阅读
本文主要讲述C++ new关键字和operator new, placement new之间的种种关联,new的底层实现,以及operator new的重载和一些在内存池,STL中的应用。
一. new operator 和 operator new
new operator:指我们在C++里通常用到的关键字,比如A* a = new A;operator new:它是一个操作符,并且可被重载(类似加减乘除的操作符重载)
关于这两者的关系,我找到一段比较经典的描述(来自于 见参考文献:
operator new can be called explicitly as a regular function, but in C++, new is an operator with a very specific behavior: An expression with the new operator, first calls function operator new (i.e., this function) with the size
of its type specifier as first argument, and if this is successful, it then automatically initializes or constructs the object (if needed). Finally, the expression evaluates as a pointer to the appropriate type.
比如我们写如下代码:
A* a = new A;
我们知道这里分为三步:1.分配内存,2.调用A()构造对象,3. 返回分配指针。事实上,分配内存这一操作就是由operator new(size_t)来完成的,如果类A重载了operator new,那么将调用A::operator
new(size_t ),否则调用全局::operator new(size_t ),后者由C++默认提供。因此前面的步骤也就是:
调用operator new (sizeof(A))调用A:A()返回指针
这里再一次提出来是因为后面关于这两步会有一些变形,在关于placement new那里会讲到。先举个简单例子
//平台:Visual Stdio 2008
#include&iostream&
std::cout&&&call A constructor&&&std::
std::cout&&&call A destructor&&&std::
int _tmain(int argc, _TCHAR* argv[])
A* a = new A;
system(&pause&);
下面我们跟踪一下A反汇编代码,由于Debug版本反汇编跳转太多,因此此处通过Release版本在A* a = new A;处设断点反汇编:&
在Release版本中,构造函数和析构函数都是直接展开的。
A* a = new A;
1不含数据成员的类占用一字节空间,此处压入sizeof(A)
operator new (13013C2h) ;调用operator new(size_t size)
esi,返回值保存到esi
esp,4 ;平衡栈
dword ptr [esp+8],
dword ptr [esp+14h],0
esi,在operator new之后,检查其返回值,如果为空(分配失败),则不调用A()构造函数
wmain+62h (1301062h) ;为空 跳过构造函数部分
eax,dword ptr [__imp_std::endl (1302038h)] ;构造函数内部,输出字符串
ecx,dword ptr [__imp_std::cout (1302050h)]
offset string &call A constructor& (1302134h)
std::operator&&&std::char_traits&char& & (13011F0h)
dword ptr [__imp_std::basic_ostream&char,std::char_traits&char& &::operator&& (1302040h)]
wmain+64h (1301064h) ;构造完成,跳过下一句
esi,将esi置空,这里的esi即为new A的返回值
dword ptr [esp+14h],0FFFFFFFFh
esi,检查a是否为空
wmain+9Bh (130109Bh) ;如果为空,跳过析构函数和operator delete
edx,dword ptr [__imp_std::endl (1302038h)] ;析构函数 输出字符串
eax,dword ptr [__imp_std::cout (1302050h)]
offset string &call A destructor& (1302148h)
std::operator&&&std::char_traits&char& & (13011F0h)
dword ptr [__imp_std::basic_ostream&char,std::char_traits&char& &::operator&& (1302040h)]
operator delete (13013BCh) ;调用operator delete
通过反汇编可以确认A* = new A的三个步骤,delete a类似&
,包含了~A()和operator delete(a)两个步骤。
二. operator new的三种形式
operator new有三种形式:
throwing (1)
void* operator new (std::size_t size) throw (std::bad_alloc);
nothrow (2)
void* operator new (std::size_t size, const std::nothrow_t& nothrow_value) throw();
placement (3)
void* operator new (std::size_t size, void* ptr) throw();
(1)(2)的区别仅是是否抛出异常,当分配失败时,前者会抛出bad_alloc异常,后者返回null,不会抛出异常。它们都分配一个固定大小的连续内存。
A* a = new A; //调用throwing(1)
A* a = new(std::nothrow) A; //调用nothrow(2)
(3)是placement new,它也是对operator new的一个重载,定义于#include &new&中,它多接收一个ptr参数,但它只是简单地返回ptr。其在new.h下的源代码如下:
#ifndef __PLACEMENT_NEW_INLINE
#define __PLACEMENT_NEW_INLINE
inline void *__cdecl operator new(size_t, void *_P)
{return (_P); }
_MSC_VER &= 1200
inline void __cdecl operator delete(void *, void *)
那么它究竟有什么用呢?事实上,它可以实现在ptr所指地址上构建一个对象(通过调用其构造函数),这在内存池技术上有广泛应用。&
它的调用形式为:
new(p) A(); //也可用A(5)等有参构造函数
placement new本身只是返回指针p,new(p) A()调用placement new之后,还会在p上调用A:A(),这里的p可以是动态分配的内存,也可以是栈中缓冲,如char buf[100]; new(buf) A();&
我们仍然可以通过一个例子来验证:
#include &iostream&
std::cout&&&call A constructor&&&std::
std::cout&&&call A destructor&&&std::
int _tmain(int argc, _TCHAR* argv[])
A* p = (A*)::operator new(sizeof(A)); //分配
new(p) A(); //构造
::operator delete(p); //释放
system(&pause&);
上面的代码将对象的分配,构造,析构和释放分离开来,这也是new和delete关键字两句就能完成的操作。&
先直接运行可以看到程序输出:
call A constructor
call A destructor
再分别注释掉new(a) A();和a-&~A();两句,可以看到对应的构造和析构函数将不会被调用。
然后查看反汇编:
//平台: Visual Studio 2008 Debug版
A* a = (A*)::operator new(sizeof(A)); //分配
operator new (0F91208h) ;调用::operator new(size_t size)也就是throwing(1)版本
dword ptr [ebp-14h],返回地址放入[ebp-14h] 即为p
new(a) A(); //构造
eax,dword ptr [ebp-14h]
operator new (0F91280h);调用operator new(size_t, void* p)即placement(3)版本 只是简单返回p
dword ptr [ebp-0E0h],将p放入[ebp-0E0h]
dword ptr [ebp-4],0
dword ptr [ebp-0E0h],0判断p是否为空
wmain+81h (0F91561h)如果为空 跳过构造函数
ecx,dword ptr [ebp-0E0h] ;取出p到ecx
A::A (0F91285h)调用构造函数 根据_thiscall调用约定 this指针通过ecx寄存器传递
dword ptr [ebp-0F4h],将返回值(this指针)放入[ebp-0F4h]中
wmain+8Bh (0F9156Bh)跳过下一句
dword ptr [ebp-0F4h],0将[ebp-0F4h]置空 当前面判断p为空时执行此语句
ecx,dword ptr [ebp-0F4h] ;[ebp-0F4h]为最终构造完成后的this指针(或者为空) 放入ecx
dword ptr [ebp-0ECh],又将this放入[ebp-0ECh] 这些都是调试所用
dword ptr [ebp-4],0FFFFFFFFh
ecx,dword ptr [ebp-14h] ;从[ebp-14h]中取出p
A::`scalar deleting destructor' (0F91041h) ;调用析构函数(跟踪进去比较复杂 如果在Release下,构造析构函数都是直接展开的)
::operator delete(a); //释放
eax,dword ptr [ebp-14h]将p放入eax
operator delete (0F910B9h);调用operator delete(void* )
从反汇编中可以看出,其实operator new调用了两次,只不过每一次调用不同的重载函数,并且placement new的主要作用只是将p放入ecx,并且调用其构造函数。&
事实上,在指定地址上构造对象还有另一种方法,即手动调用构造函数:p-&A::A();&这里要加上A::作用域,用p-&A::A();替换掉new(p)
A();仍然能达到同样的效果,反汇编:
A* a = (A*)::operator new(sizeof(A)); //分配
operator new (1061208h)
dword ptr [a],eax
//new(a) A();
a-&A::A();
ecx,dword ptr [a]
operator new (1061285h)
ecx,dword ptr [a]
A::`scalar deleting destructor' (1061041h)
::operator delete(a); //释放
eax,dword ptr [a]
operator delete (10610B9h)
比之前的方法更加简洁高效(不需要调用placement new)。不知道手动调用构造函数是否有违C++标准或有什么隐晦,我在其他很多有名的内存池(包括SGI STL alloc)实现上看到都是用的placement new,而不是手动调用构造函数。
三. operator new重载
前面简单提到过A* p = new A;所发生的事情:先调用operator new,如果类A重载了operator new,那么就使用该重载版本,否则使用全局版本::operatro
new(size_t size)。
上面提到的throwing(1)和nothrow(2)的operator new是可以被重载的,比如:
#include &iostream&
std::cout&&&call A constructor&&&std::
std::cout&&&call A destructor&&&std::
void* operator new(size_t size)
std::cout&&&call A::operator new&&&std::
return malloc(size);
void* operator new(size_t size, const std::nothrow_t& nothrow_value)
std::cout&&&call A::operator new nothrow&&&std::
return malloc(size);
int _tmain(int argc, _TCHAR* argv[])
A* p1 = new A;
delete p1;
A* p2 = new(std::nothrow) A;
delete p2;
system(&pause&);
运行结果:
call A::operator new
call A constructor
call A destructor
call A::operator new nothrow
call A constructor
call A destructor
如果类A中没有对operator new的重载,那么new A和new(std::nothrow) A; 都将会使用全局operator new(size_t size)。可将A中两个operator new注释掉,并且在A外添加一个全局operator new重载:
void* ::operator new(size_t size)
std::cout&&&call global operator new&&&std::
return malloc(size);
程序输出:
call global operator new
call A constructor
call A destructor
call global operator new
call A constructor
call A destructor
注意,这里的重载遵循作用域覆盖原则,即在里向外寻找operator new的重载时,只要找到operator new()函数就不再向外查找,如果参数符合则通过,如果参数不符合则报错,而不管全局是否还有相匹配的函数原型。比如如果这里只将A中operator new(size_t,
const std::nothrow_t&)删除掉,就会报错:
error C2660: “A::operator new”: 函数不接受 2 个参数。
对operator new的重载还可以添加自定义参数,如在类A中添加
void* operator new(size_t size, int x, int y, int z)
std::cout&&&X=&&&x&&&
Y=&&&y&&& Z=&&&z&&std::
return malloc(size);
这种重载看起来没有什么大作用,因为它operator new需要完成的任务只是分配内存,但是通过对这类重载的巧妙应用,可以让它在动态分配内存调试和检测中大展身手。这将在后面operator new重载运用技巧中展现。
至于placement new,它本身就是operator new的一个重载,不需也尽量不要对它进行改写,因为它一般是搭配 new(p) A(); 工作的,它的职责只需简单返回指针。
四. operator new运用技巧和一些实例探索
1. operator new重载运用于调试:
前面提到如何operator new的重载是可以有自定义参数的,那么我们如何利用自定义参数获取更多的信息呢,这里一个很有用的做法就是给operator new添加两个参数:char* file, int line,这两个参数记录new关键字的位置,然后再在new时将文件名和行号传入,这样我们就能在分配内存失败时给出提示:输出文件名和行号。
那么如何获取当前语句所在文件名和行号呢,windows提供两个宏:__FILE__和__LINE__。利用它们可以直接获取到文件名和行号,也就是 new(__FILE__, __LINE__) 由于这些都是不变的,因此可以再定义一个宏:#define new new(__FILE__, __LINE__)。这样我们就只需要定义这个宏,然后重载operator new即可。
源代码如下,这里只是简单输出new的文件名和行号。
std::cout&&&call A constructor&&&std::
std::cout&&&call A destructor&&&std::
void* operator new(size_t size, const char* file, int line)
std::cout&&&call A::operator new on file:&&&file&&&
line:&&&line&&std::
return malloc(size);
return NULL;
//Test.cpp
#include &iostream&
#include &A.h&
#define new new(__FILE__, __LINE__)
int _tmain(int argc, _TCHAR* argv[])
A* p1 = new A;
delete p1;
system(&pause&);
call A::operator new on file:d:\desktop\test\test.cpp line:8
call A constructor
call A destructor
注意:需要将类的声明实现与new的使用隔离开来。并且将类头文件放在宏定义之前。否则在类A中的operator new重载中的new会被宏替换,整个函数就变成了:void* operator new(__FILE__, __LINE__)(size_t size, char*
file, int line),编译器自然会报错。
2. 内存池优化
operator new的另一个大用处就是内存池优化,内存池的一个常见策略就是分配一次性分配一块大的内存作为内存池(buffer或pool),然后重复利用该内存块,每次分配都从内存池中取出,释放则将内存块放回内存池。在我们客户端调用的是new关键字,我们可以改写operator new函数,让它从内存池中取出(当内存池不够时,再从系统堆中一次性分配一块大的),至于构造和析构则在取出的内存上进行,然后再重载operator delete,它将内存块放回内存池。关于内存池和operator
new在参考文献中有一篇很好的文章。这里就不累述了。
3. STL中的new
在SGI STL源码中,defalloc.h和stl_construct.h中提供了最简单的空间配置器(allocator)封装,见《STL源码剖析》P48。它将对象的空间分配和构造分离开来,虽然在defalloc.h中仅仅是对::operator new和::operator delete的一层封装,但是它仍然给STL容器提供了更加灵活的接口。SGI STL真正使用的并不是defalloc.h中的分配器,而是stl_alloc.h中的SGI精心打造的&双层级配置器&,它将内存池技术演绎得淋漓尽致,值得细细琢磨。顺便提一下,在stl_alloc.h中并没有使用::operator
new/delete 而直接使用malloc和free。具体缘由均可参见《STL源码剖析》。
五. delete的使用
delete的使用基本和new一致,包括operator delete的重载方式这些都相似,只不过它的参数是void*,返回值为void。但是有一点需要注意,operator delete的自定义参数重载并不能手动调用。比如
void* operator new(size_t size, int x)
cout&&& x = &&&x&&
return malloc(size);
void operator delete(void* p, int x)
cout&&& x = &&&x&&
如下调用是无法通过的:
A* p = new(3) A;//ok
delete(3)//error C2541: “delete”: 不能删除不是指针的对象
那么重载operator delete有什么作用?如何调用?事实上以上自定义参数operator delete 只在一种情况下被调用:当new关键字抛出异常时。
可以这样理解,只有在new关键字中,编译器才知道你调用的operator new形式,然后它会调用对应的operator delete。一旦出了new关键字,编译器对于这块内存是如何分配的一无所知,因此它只会调用默认的operator delete,而至于为什么不能主动调用自定义delete(而只能老老实实delete p),这个就不知道了。
细心观察的话,上面operator new用于调试的例子代码中,由于我们没有给出operator new对应的operator delete。在VS2008下会有如下警告:
warning C4291: “void *A::operator new(size_t,const char *,int)”: 未找到匹配的删除运算符;如果初始化引发异常,则不会释放内存
六. 关于new和内存分配的其他
1. set_new_handler
还有一些零散的东西没有介绍到,比如set_new_handler可以在malloc(需要调用set_new_mode(1))或operator
new内存分配失败时指定一个入口函数new_handler,这个函数完成自定义处理(继续尝试分配,抛出异常,或终止程序),如果new_handler返回,那么系统将继续尝试分配内存,如果失败,将继续重复调用它,直到内存分配完毕或new_handler不再返回(抛出异常,终止)。下面这段程序完成这个测试:
#include &iostream&
#include &new.h&// 使用_set_new_mode和set_new_handler
void nomem_handler()
std::cout&&&call nomem_handler&&&std::
int main()
_set_new_mode(1);
//使new_handler有效
set_new_handler(nomem_handler);//指定入口函数 函数原型void f();
std::cout&&&try to alloc 2GB memory....&&&std::
char* a = (char*)malloc(2*24);
std::cout&&&ok...I got it&&&std::
system(&pause&);
程序运行后会一直输出call nomem_handler 因为函数里面只是简单输出,返回,系统尝试分配失败后,调用nomem_handler函数,由于该函数并没有起到实际作用(让可分配内存增大),因此返回后系统再次尝试分配失败,再调用nomem_handler,循环下去。&
在SGI STL中的也有个仿new_handler函数:oom_malloc
2. new分配数组
new[]和new类似,仍然会优先调用类中重载的operator new[]。另外还要注意的是,在operator new[](size_t size)中传入的并不是sizeof(A)*3。而要在对象数组的大小上加上一个额外数据,用于编译器区分对象数组指针和对象指针以及对象数组大小。在VS2008(32 bit)下这个额外数据占4个字节,一个int大小。测试代码如下
std::cout&&&call A constructor&&&std::
std::cout&&&call A destructor&&&std::
void* operator new[](size_t size)
std::cout&&&call A::operator new[] size:&&&size&&std::
return malloc(size);
void operator delete[](void* p)
std::cout&&&call A::operator delete[]&&&std::
void operator delete(void* p)
//Test.cpp
#include &iostream&
#include &A.h&
void* operator new[](size_t size)
std::cout&&&call global new[] size: &&&size&&std::
return malloc(size);
void operator delete[](void* p)
std::cout&&&call global delete[] &&&std::
int _tmain(int argc, _TCHAR* argv[])
std::cout&&&sizeof A &&&sizeof(A)&&std::
A* p1 = new A[3];
delete []p1;
system(&pause&);
sizeof A 1
call global new[] size: 7
call A constructor
call A constructor
call A constructor
call A destructor
call A destructor
call A destructor
call A::operator delete[]
简单跟踪了一下,operator new[]返回的是0x005b668 而最后new关键字返回给p的是0x005b66c。也就是说p就是数组的起始地址,这样程序看到的内存就是线性的,不包括前面的额外数据。
在内存中,可以看到前面的四个字节额外数据是0x 也就是3,代表数组元素个数。后面三个cd是堆在Debug中的默认值(中文的cdcd就是”屯”,栈的初始值为cc,0xcccc中文”烫”)。再后面的0xfdfdfdfd应该是堆块的结束标志,前面我有博客专门跟踪过。
注:其实在malloc源码中也有内存池的运用,而且也比较复杂。最近在参考dlmalloc版本和STL空间适配器,真没有想到一个内存分配能涉及这么多的东西。
七. 参考文献:
1.&operator new的三种形式&
2.&c++ operator new重载和内存池技术&
3.《STL源码剖析》 空间配置器&
4.&&一篇关于理解C++ New的好文&
5.&&陈硕的Blog
转载请注明出处:
&&相关文章推荐
* 以上用户言论只代表其个人观点,不代表CSDN网站的观点或立场
访问:155812次
积分:1779
积分:1779
排名:千里之外
原创:53篇
评论:29条
(1)(1)(1)(2)(3)(3)(4)(3)(13)(22)
(window.slotbydup = window.slotbydup || []).push({
id: '4740887',
container: s,
size: '250,250',
display: 'inlay-fix'问一个比较小白的问题,java中for循环里面new一个数组,内存是线性增加的吗? - 知乎13被浏览1639分享邀请回答0添加评论分享收藏感谢收起一、Effective C++条款16:成对使用new和delete时采用相同的形式
通常我们使用new和delete有两种情形,第一,动态的为单一对象分配内存,第二,动态的创建数组。new和delete的使用需遵循许多规则,这里着重理解相同形式这一关键词。且看下面的动作有什么错?
std::string *stringArray = new std::string[100];
delete stringA
乍看起来,同时使用了new和delete,似乎没什么错误。这里使用new在运行阶段动态分配了100string对象,但是仅使用delete只释放掉了单一对象的所占用的内存,其中99个string对象不太可能被适当的删除,因为它们的析构函数没有被调用。
请注意:这里仅仅同时使用了new和delete,但并没有满足相同的形式这一条件。我们所面临的问题可以更简单的叙述为:即将被删除的那个指针,所指的是单一对象还是对象数组?另外,请尽量使用,以防止写代码时忘记释放掉new分配的内存而导致内存泄漏。
如果在new表达式中使用了[],必须在相应的delete表达式中使用[]。如果在new表达式中不使用[],一定不要在相应的delete表达式中使用[]。确保“同时使用、相同形式”。
二、使用new动态分配内存
为单一对象(可以是数据结构,基本类型或类)获得并指定分配内存的通用格式如下:
typeName *pointter_name &= new typeName;
例如,在运行阶段为一个int值分配未命名的内存,并使用指针来访问这个值,可以写为:int * pn =
程序员告诉new要为int类型的数据类型分配内存,new根据这种数据类型找到长度(在这里int为4个字节)正确的内存块,并返回内存块的地址,程序员负责将该地址赋给一个指针(pn是被声明为指向int的指针,现在pn是一个地址,而*pn为存储在那里的值)。
请看下面的例子:
*pt = 1001;
double *pd =
*pd = 3.14;
则有:如果这里int占4个字节,double占8个字节,则sizeof(pt)为4,sizeof(*pt)为4。sizeof(pd)为4,sizeof(*pd)为8。这里,指向int的指针的长度与指向double的指针相同,它们都是地址,但由于声明了指针的类型,因此程序知道*pd为8个字节的double值,*pt为4个字节的int值。
三、使用new和delete时要遵循的规则
不要使用delete释放不是new分配的内存。不要使用delete释放同一个内存两次。使用new[]为数组分配的内存一定要使用delete[]释放,不能使用delete;new为单个对象分配的内存一定使用delete来释放。对空指针delete是安全的。
四、使用动态数组
创建动态数组后,如何使用数组中的元素?
int *pt = new int[10]; // get &a block of 10 ints
如果这里int占4个字节,上述语句中sizeof(pt)为4个字节,sizeof(*pt)为4 × 10个字节。那么如何使用数组中的元素呢?其实和普通数组一样。
double *pd = new double[3];
pd[0] = 1.0;
pd[1] = 2.0;
pd[2] = 3.0;
那么,pd[0](或*pd) 的值为1.0。如果有pd = pd + 1;, 那么pd[0]的值为2.0。另外请注意,不能修改数组名(常量),但指针(pd)为变量,因此可以修改它的值。而且pd+1的效果是指指针指向数组第一个元素的下一个位置,即第二个元素,因此pd[0]为2.0。
指针和数组等价的原因在于指针算术和C++内部处理数组的方式。首先我们来看一下算术,将整型变量加1后,其值将增加1;但将指针变量增加1后,增加的量等于它指向的数据类型的字节数,如:将指向double的指针加1后,如果系统对double使用8个字节存储,则其值增加8。另外,C++将数组名解释为数组第一个元素的地址。
double arrD[3] = {1.0, 2.0, 3.0};
short arrSh[3] = {1, 2, 3};
double *pd = arrD; // 等价于&arrD[0]
short *psh = &arrSh[0];
假设double占4个字节,short占2个字节,则:arrD[0] ,*arrD和*(arrD + 0)等价,其值为1;*psh为1。
sizeof(arrD)为3 × 8 = 24,sizeof(pd)为4,sizeof(*pd)为8。
sizeof(arrSh)为3 × 2 = 6,sizeof(psh)为4,sizeof(*psh)为2。
这是因为&指针为指向变量的地址,且地址占4个字节;对指针的解引用为指针指向变量的值,所占内存谓该值所属类型所占的字节数。
注意:虽然数组名被解释为指向数组第一个元素的地址,但不能修改数组名,是因为数组名为常量;可以修改指针,是因为指针(指向变量的地址)为变量。
pointerName = pointerName + 1; // valid
arrayName = arrayName + 1; // not alllowed
数组的地址
数组名被解释为其第一个元素的地址,而对数组名应用地址运算符时,得到的是整个数组的地址。
short tell[10]; & & // tell为20字节的数组
cout && & &// &tell[0] 即数组第一个元素的地址,占2个字节的内存块的地址
cout && & &// &整个数组的地址,从数字上说,该地址和tell的地址相同,但是&tell为占20个字节的内存块的地址
总之,使用new来创建数组以及使用指针来访问不同的元素时,只需要把指针当成数组名对待即可。
本文已收录于以下专栏:
相关文章推荐
代码写多了,就麻木了。new和delete很好用,平时用的时候没想太多。但如果“想太多”就会引发出很多东西。
new和delete跟sizeof一样,是操作符,关键字,而不是函数。new和dele...
C++学习:malloc/free和new/delete简介:
在C语言中,内存的申请和释放采用的是malloc/calloc/realloc/free来实现的,而在C++语言中,不仅可以使用以上的...
原博地址:内存分配方式,堆区,栈区,new/delete/malloc/free
1.内存分配方式
内存分配方式有三种:
[1]从静态存储区域分配。内存在程序编译的时候就已经分配好,这块内存...
1.内存分配方式内存分配方式有三种:[1]从静态存储区域分配。内存在程序编译的时候就已经分配好,这块内存在程序的整个运行期间都存在。例如全局变量,static变量。[2]在栈上创建。在执行函数时,函数...
C++ 方式的内存分配与释放 new 和 delete
在内存管理上,C++ 和 C 有着完全不同的两套方案。当然,C++的总是同时兼容C。C的那一套方案在C++里同样可行。
我们首先...
NewDeleteDemo.cpp
/***********************************...
C语言里面的动态内存分配用的是malloc()/calloc()和free()其中,malloc用于分配一个单位的内存,返回指针。例如malloc(sizeof int)
calloc用于分配多个内...
内存分配方式 内存分配方式有三种: [1] 从静态存储区域分配。内存在程序编译的时候就已经分配好,这块内存在程序的整个运行期间都存在。例如全局变量, static 变量。 [2] 在栈上创建。在执行函...
一、动态内存分配的引入
一般,变量和对象在内存中的分配都是编译器在编译程序时安排好了的,这带来了极大的不便,如数组必须大开小用,指针必须指向一个已经存在的变量或对象。动态内存分配解决了这个问题。
他的最新文章
讲师:王哲涵
讲师:王渊命
您举报文章:
举报原因:
原文地址:
原因补充:
(最多只允许输入30个字)

我要回帖

更多关于 new 一个数组 的文章

 

随机推荐