1、为什么要使用动态数组?
在实际的编程中,往往会发生这种情况,即所需的内存空间取决于实际输入的数据,而无法预先确定。对于这种问题,用静态数组的办法很难解决。为解决上述问题,C语言提供了一些内存管理函数,这些内存管理函数结合指针可以按需要动态地分配内存空间,来构建动态数组,也可把不再使用的空间回收待用,为有效地利用内存资源提供了手段。
2、int a[10]与int *a=new int[10]的区别
int a[10];//系统在栈中分配空间,系统会自动实现内存的分配和回收。 int *a = new int[10];//系统在堆中分配空间,不用时需要使用delete[] a进行内存释放;如果程序中没有delete,可能会造成内存的泄露。
栈的实际内存是连续内存,因此可分配空间较小,堆可以是非连续内存,因此可以分配较大内存。因此,如果需要分配较大内存,需要分配在堆上;使用int a[10]这种方式,内存大小需要用常量指定,比如这里的10。不能用int m=10;int a[m]这种方式。但是int* a= new这种方式可以,因此在动态分配内存上,后者有非常大的优势。
3、malloc和new的区别是什么?
1
2
3
|
int
*p;
p =
new
int
;
//返回类型为int* 类型(整数型指针),分配大小为 sizeof(int);
|
1
2
3
|
int
* parr;
parr =
new
int
[100];
//返回类型为 int* 类型(整数型指针),分配大小为 sizeof(int) * 100;
|
1
2
3
4
5
6
7
|
int
* p;
p = (
int
*)
malloc
(
sizeof
(
int
)*128);
//分配128个(可根据实际需要替换该数值)整型存储单元,
//并将这128个连续的整型存储单元的首地址存储到指针变量p中
double
*pd=(
double
*)
malloc
(
sizeof
(
double
)*12);
//分配12个double型存储单元,
//并将首地址存储到指针变量pd中
|
1
|
int
* p = (
int
*)
malloc
(1);
|
什么是堆:堆是大家共有的空间,分全局堆和局部堆。全局堆就是所有没有分配的空间,局部堆就是用户分配的空间。堆在操作系统对进程 初始化的时候分配,运行过程中也可以向系统要额外的堆,但是记得用完了要还给操作系统,要不然就是内存泄漏。
什么是栈:栈是线程独有的,保存其运行状态和局部自动变量的。栈在线程开始的时候初始化,每个线程的栈互相独立。每个函数都有自己的栈,栈被用来在函数之间传递参数。操作系统在切换线程的时候会自动的切换栈,就是切换SS/ESP寄存器。栈空间不需要在高级语言里面显式的分配和释放。
栈是由编译器自动分配释放,存放函数的参数值、局部变量的值等。操作方式类似于数据结构中的栈。
堆一般由程序员分配释放,若不释放,程序结束时可能由OS回收。注意这里说是可能,并非一定。所以我想再强调一次,记得要释放!注意它与数据结构中的堆是两回事,分配方式倒是类似于链表。举个例子,如果你在函数上面定义了一个指针变量,然后在这个函数里申请了一块内存让指针指向它。实际上,这个指针的地址是在栈上,但是它所指向的内容却是在堆上面的!比如说下面这个函数:
// code...
void Function(void)
{ char *p = (char *)malloc(100 * sizeof(char)); }
就这个例子,千万不要认为函数返回,函数所在的栈被销毁指针也跟着销毁,申请的内存也就一样跟着销毁了!这绝对是错误的!因为申请的内存在堆上,而函数所在的栈被销毁跟堆完全没有啥关系。所以,还是那句话:记得释放!
6、free()到底释放了什么
free()释放的是指针指向的内存!注意!释放的是内存,不是指针!这点非常非常重要!指针是一个变量,只有程序结束时才被销毁。释放了内存空间后,原来指向这块空间的指针还是存在!只不过现在指针指向的内容的垃圾,是未定义的,所以说是垃圾。因此,前面我已经说过了,释放内存后把指针指向NULL,防止指针在后面不小心又被解引用了。非常重要啊这一点!
例子:
头文件:#include<stdlib.h>建议加上#include<malloc.h>就不需要stdlib了
具体实现:类型+指针=(类型*)calloc(数组大小,sizeof(类型));
动态创建一维数组的例子:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
#include<stdio.h>
#include<stdlib.h>//
#include<malloc.h>//这两个头文件任选一个就行了,上面的范围更大
int
main()
{
int
n,i;
scanf
(
"%d"
,&n);
//输入数组大小
int
*p=(
int
*)
calloc
(n,
sizeof
(
int
));
//申请了一个长度为n的数组,用p指向首地址
for
(i=0;i<n;i++)
//位数组元素赋值
{
scanf
(
"%d"
,p+i);
}
for
(i=0;i<n;i++)
//输出数组元素
{
printf
(
"%d "
,*(p+i));
}
free
(p);
//释放申请的内存空间
return
0;
}
|
输入:
2
1
2
输出:12
注:当用指针指向动态数组时,不能用sizeof确定这个数组中元素的个数。
例:
int *p=(int *)malloc(n);//4 sizeof(p)得到的结果是P指针类型,即:int型指针占内存的大小。
int sn[5]={1,2};///20 sizeof(sn)得到的是静态数组sn占内存大小。
计算数组sn中元素的个数的方法:
//计数法
int i=0;
while(sn[i++]!=0);//有效元素个数为i-1
//sizeof()
len=sizeof(sn)/sizeof(int);//数组长度
动态创建二位数组的例子:
///动态创建二维数组
#include <stdlib.h>
#include <stdio.h>
int main()
{
int n1,n2;
int **array,i,j;
puts("输入一维长度:");
scanf("%d",&n1);
puts("输入二维长度:");
scanf("%d",&n2);
array=(int**)malloc(n1*sizeof(int*)); //第一维
for(i=0;i<n1; i++)
{
array[i]=(int*)malloc(n2* sizeof(int));//第二维
for(j=0;j<n2;j++)
{
array[i][j]=i+j+1;
printf("%d\t",array[i][j]);
}
puts("");
}
for(i=0;i<n1;i++)
{
free(array[i]);//释放第二维指针
}
free(array);//释放第一维指针
return 0;
}
http://write.blog.csdn.net/postedit/51351183
http://www.linuxidc.com/Linux/2016-01/127591.htm
http://www.cnblogs.com/fly1988happy/archive/2012/04/26/2470542.html