C语言动态内存开辟

动态内存开辟

    • 1.malloc、free
      • 1.1 malloc的原型为:
      • 1.2 malloc的用法:
      • 1.3 free的原型:
      • 1.4 free的用法:
    • 2.calloc
      • 2.1 calloc原型:
      • 2.2 calloc用法:
    • 3.realloc
      • 3.1 realloc原型:
      • 3.2 realloc用法:

1.malloc、free

1.1 malloc的原型为:

void *malloc( size_t size );

1.2 malloc的用法:

看下面代码:

int* p=(int*)malloc(10*sizeof(int));

上面代码意思是开辟10个int类型大小的空间,将其强制转换为int型,并用int类型指针接收。
1.若开辟成功,则返回开辟空间的头指针。
2.若开辟失败,则返回空指针。
所以,用了malloc后一定要判断!
如下面代码:

#include
#include// malloc 所在头文件
#include// errno,h 错误码库
int main()       // 与 函数 strerror(错误信息报告)  搭配使用,将错误码转化为错误信息,并返回它的地址
{
	// 我想向内存申请 10个 整形的 空间
	int* p = (int*)malloc(/*INT_MAX*/10 * (sizeof(int)));// 因为  malloc 函数 返回的是 万能*(无类型)指针,所以需要转化类型
	// malloc 函数 也有 开辟空间失败的时候,
	// 比如 内存 只有 4 个 g ,我要开辟 8 g 空间,这肯定是会失败的,返回 空指针 NULL
	
	if (p == NULL)
	{
		// 找出打印错误原因的方式
		printf("%s\n", strerror(errno));// malloc(INT_MAX)
		                //输出 Not enough space
	}
	else
	{
		// 正常使用 空间
		int i = 0;
		for (i = 0; i < 10; i++)
		{
			*(p + i) = i;// 赋值 0 1 2 3 4 5 6 7 8 9
		}
		for (i = 0; i < 10; i++)
		{
			printf("%d ", *(p + i));// 输出 0 1 2 3 4 5 6 7 8 9
		}
		printf("\n");
	}
	// 当动态申请的空间 不再使用的时候
	// 就会把空间 还给 操作系统
	// 这时候 就会用到 free 函数
	return 0;
}

而malloc的使用需和free一起,即:用malloc开辟一处空间,必须由free来释放。否则会造成内存泄露!

1.3 free的原型:

void free( void *memblock );

1.4 free的用法:

free专门用于动态内存的释放和回收。如下代码:

#include
#include// malloc,free 所在头文件
#include
int main()
{
	// 我想向内存申请 10个 整形的 空间
	int* p = (int*)malloc(10 * (sizeof(int)));
	if (p == NULL)
	{
		// 找出打印错误原因的方式
		printf("%s\n", strerror(errno));// malloc(INT_MAX)
		                //输出 Not enough space
	}
	else
	{
		// 正常使用 空间
		int i = 0;
		for (i = 0; i < 10; i++)
		{
			*(p + i) = i;// 0 1 2 3 4 5 6 7 8 9
		}
		for (i = 0; i < 10; i++)
		{
			printf("%d ", *(p + i));// 0 1 2 3 4 5 6 7 8 9
		}
		printf("\n");
	}
	// 当动态申请的空间 不再使用的时候
	// 就会把空间 还给 操作系统
	// 这时候 就会用到 free 函数
	free(p);// 用完了再释放,也就是说 在打印完之后(调用完之后),再释放(p的值没有改变,还可以通过它找到该空间)
	//  虽然当前空间 不属于当前程序,但是依然 可以通过 p  找到  该空间,很有可能会破坏这个空间
	//  所以先在这个 指针 依然很危险

	p = NULL; // free函数 释放完空间后,不会改 p 的值,所以我们自己主动改
	return 0;
}
// 最后请注意:free 是用来 释放 动态开辟的内存

// free 函数 的 特殊情况:
//1. 如果参数 ptr 指向的空间不是动态开辟的,那 free 函数的行为是未定义的
//2. 如果参数 ptr 是 NULL 指针,则函数 什么 都不做。

1.首先,将动态开辟的空间free掉。
2.再将free掉的空间的头指针制为空即可。

2.calloc

2.1 calloc原型:

void *calloc( size_t num, size_t size );

2.2 calloc用法:

calloc和malloc一样,都用于动态内存开辟,区别点在于calloc具有初始化功能,即:calloc可将开辟的空间进行初始化操作。
如下代码:

 用法跟 malloc 函数差不多
#include
#include
#include
int main()
{                     // 开一块空间,空间大小为 10 * int == 40 字节
	int* p = (int*)calloc(10, sizeof(int));// 比 malloc 函数 多一个元素个数 参数
	if (p == NULL)    
	{
		printf("%s\n", strerror(errno));
	}
	else
	{
		int i = 0;
		for (i = 0; i < 10; i++)// 因为 p 为整形指针变量,指向一个 int 类型 元素,所以有 10个 int 元素
		{
			printf("%d ", *(p + i));// 0 0 0 0 0 0 0 0 0 0
		}  //calloc 在开辟好空间后,会把空间的所有字节内容全部初始化为 零
	}
	free(p);
	p = NULL;// 狗头拿来!!
	return 0;
}

3.realloc

3.1 realloc原型:

void *realloc( void *memblock, size_t size );

3.2 realloc用法:

有时我们会发现过去申请的空间太小了,有时候我们又会觉得申请的空间过大了,为了合理的使用内存,我们一定会对内存的大小做灵活的调整,realloc函数就可以做到对动态开辟内存大小的调整。
如以下代码:

#include 
#include 
 
int main(int argc, char* argv[], char* envp[])
{
	int input;
	int n;
	int *numbers1;
	int *numbers2;
	numbers1=NULL;
 
	if((numbers2=(int *)malloc(5*sizeof(int)))==NULL)//为numbers2在堆中分配内存空间
	{
		printf("malloc memory unsuccessful");
		exit(1);
	}
	
	printf("numbers2 addr: %8X\n",(int)numbers2);
 
	for(n=0;n<5;n++) //初始化
	{
		*(numbers2+n)=n;
		//printf("numbers2's data: %d\n",*(numbers2+n));
	}
 
	printf("Enter new size: ");
	scanf("%d",&input);
 
	//重新分配内存空间,如果分配成功的话,就释放numbers2指针,
	//但是并没有将numbers2指针赋为NULL,也就是说释放掉的是系统分配的堆空间,
	//和该指针没有直接的关系,现在仍然可以用numbers2来访问这部分堆空间,但是
	//现在的堆空间已经不属于该进程的了。
	numbers1=(int *)realloc(numbers2,(input+5)*sizeof(int));
 
	if(numbers1==NULL)
	{
		printf("Error (re)allocating memory");
		exit(1);
	}
	
	printf("numbers1 addr: %8X\n",(int)numbers1);
 
	/*for(n=0;n<5;n++) //输出从numbers2拷贝来的数据
	{
		printf("the numbers1's data copy from numbers2: %d\n",*(numbers1+n));
	}*/
 
	for(n=0;n<input;n++)//新数据初始化
	{
		*(numbers1+5+n)=n+5;
		//printf("numbers1' new data: %d\n",*(numbers1+5+n));
	}
 
	printf("\n");
 
	free(numbers1);//释放numbers1,此处不需要释放numbers1,因为在realloc()时已经释放
	numbers1=NULL;
	//free(numbers2);//不能再次释放
	return 0;
}

1.如果当前内存段后有足够的空间,realloc()返回原来的指针。
2.如果当前内存段后没有足够的空间,realloc()返回一个新的内存段的指针

你可能感兴趣的:(C语言学习笔记,c语言,动态内存开辟)