【C语言】malloc和free

【C语言】malloc和free

malloc

函数原型

void *malloc(long NumBytes);

说明:该函数分配了NumBytes个字节,并返回了指向这块内存的指针。如果分配失败,则返回一个空指针(NULL)。关于分配失败的原因,应该有多种,比如说空间不足就是一种。

free

函数原型

void free(void *FirstByte);

说明:该函数是将之前用malloc分配的空间还给程序或者是操作系统,也就是释放了这块内存,让它重新得到自由。

简单例子

char *Ptr = NULL; 
Ptr = (char *)malloc(100 * sizeof(char)); 
if (NULL == Ptr) 
{
  exit (1); 
  } 
gets(Ptr); 

free(Ptr); 
Ptr = NULL; 

当然,具体情况要具体分析以及具体解决。比如说,你定义了一个指针,在一个函数里申请了一块内存然后通过函数返回传递给这个指针,那么也许释放这块内存这项工作就应该留给其他函数了。

注意事项

A、申请了内存空间后,必须检查是否分配成功。

B、当不需要再使用申请的内存时,记得释放;释放后应该把指向这块内存的指针指向NULL,防止程序后面不小心使用了它。

C、这两个函数应该是配对。如果申请后不释放就是内存泄露;如果无故释放那就是什么也没有做。释放只能一次,如果释放两次及两次以上会

出现错误(释放空指针例外,释放空指针其实也等于啥也没做,所以释放空指针释放多少次都没有问题)。

D、虽然malloc()函数的类型是(void *),任何类型的指针都可以转换成(void *),但是最好还是在前面进行强制类型转换,因为这样可以躲过一些编译器的检查。

malloc()到底从哪里得来了内存空间?

A、malloc()到底从哪里得到了内存空间?答案是从堆里面获得空间。也就是说函数返回的指针是指向堆里面的一块内存。操作系统中有一个记录空闲内存地址的链表。当操作系统收到程序的申请时,就会遍历该链表,然后就寻找第一个空间大于所申请空间的堆结点,然后就将该结点从空闲结点链表中删除,并将该结点的空间分配给程序。

B、在使用malloc()分配内存空间后,一定要记得释放内存空间,否则就会出现内存泄漏。

free()到底释放了什么?

free()释放的是指针指向的内存!注意!释放的是内存,不是指针!指针并没有被释放,指针仍然指向原来的存储空间。指针是一个变量,只有程序结束时才被销毁。释放了内存空间后,原来指向这块空间的指针还是存在!只不过现在指针指向的内容的垃圾,是未定义的,所以说是垃圾。因此,释放内存后把指针指向NULL,防止指针在后面不小心又被解引用了。

malloc()以及free()的机制

事实上,仔细看一下free()的函数原型,也许也会发现似乎很神奇,free()函数非常简单,只有一个参数,只要把指向申请空间的指针传递给free()中的参数就可以完成释放工作!这里要追踪到malloc()的申请问题了。申请的时候实际上占用的内存要比申请的大。因为超出的空间是用来记录对这块内存的管理信息。

大多数实现所分配的存储空间比所要求的要稍大一些,额外的空间用来记录管理信息——分配块的长度,指向下一个分配块的指针等等。这就意味着如果写过一个已分配区的尾端,则会改写后一块的管理信息。这种类型的错误是灾难性的,但是因为这种错误不会很快就暴露出来,所以也就很难发现。将指向分配块的指针向后移动也可能会改写本块的管理信息。

malloc()申请的空间实际就是分了两个不同性质的空间。一个就是用来记录管理信息的空间,另外一个就是可用空间了。而用来记录管理信息的实际上是一个结构体。在C语言中,经常用结构来记录信息!下面看看这个结构体的原型:

struct mem_control_block { 
    int is_available;    //一般来说应该是一个可用空间的首地址,但这里英文单词却显示出空间是否可用的一个标记
        int size;            //这是实际空间的大小 
        };
        ```
所以,free()就是根据这个结构体的信息来释放malloc()申请的空间!而结构体的两个成员的大小我想应该是操作系统的事了。
下面看看free()的源代码

```c
void free(void *ptr)  
{ 
	struct mem_control_block *free; 
		free = ptr - sizeof(struct mem_control_block); 
			free->is_available = 1; 
				return; 
				}
				```





<think>嗯,用户问的是C语言mallocfree的使用方法。这个问题看起来基础,但对于刚接触指针内存管理的新手来说可能有点挑战。首先,我得回想一下mallocfree的基本作用。malloc用于动态分配内存,而free用于释放这些内存,防止内存泄漏。用户可能需要一个清晰的步骤说明,最好有例子。 我应该先解释为什么需要动态内存分配。比如,当数据大小在编译时不确定,比如用户输入数组的大小,这时候就需要用malloc在运行时分配。接着,需要说明malloc的函数原型,返回的是void指针,需要类型转换,还要检查是否分配成功,因为可能内存不足。 然后是free的使用,必须对应每个malloc,否则会有内存泄漏。还要提醒不要重复释放同一块内存,这会导致未定义行为。可能用户会犯的常见错误,比如忘记释放内存,或者访问已释放的内存,这些都需要提到。 接下来,结构安排可能需要分步骤,比如分配内存的步骤:计算所需空间、类型转换、检查指针。然后释放的步骤:调用free,并将指针置NULL防止悬空指针。例子部分,可以举一个动态数组的例子,比如输入n个数,动态分配数组,然后求后释放。 还要注意中英文术语的使用,比如用中文解释,但函数名如mallocfree保持英文。同时,需要确保代码示例正确,比如sizeof的正确使用,malloc的参数是字节数,可能需要乘以sizeof(int)之类的。 用户可能还关心常见问题,比如内存泄漏的后果,或者如何调试内存问题。虽然不是直接的问题,但可以简要提及,作为最佳实践。比如使用工具如Valgrind来检测内存泄漏。 另外,需要注意mallocfree必须成对出现,分配后记得释放。可能用户会混淆栈内存堆内存的区别,所以可以稍微提一下动态分配的内存在堆上,需要手动管理。 总结下来,回答的结构应该是先介绍概念,再分步骤说明使用方法,然后给出示例代码,最后指出注意事项常见错误。这样用户能有一个全面的理解,并且能按照步骤正确使用mallocfree。</think>好的,我将分步骤为您讲解C语言中`malloc`free`的使用方法: 一、malloc基础概念 1. 作用:动态分配指定字节数的堆内存 2. 函数原型:`void* malloc(size_t size);` 3. 返回值:成功返回内存块首地址,失败返回NULL 4. 必须包含头文件:`<stdlib.h>` 二、malloc使用步骤 1. 计算所需内存空间 2. 强制类型转换为目标指针类型 3. 检查是否分配成功(重要!) 示例: ```c int* arr; int n = 10; // 分配n个int类型空间 arr = (int*)malloc(n * sizeof(int)); // 必须检查指针有效性 if (arr == NULL) { printf("内存分配失败!"); exit(1); } ``` 三、free使用规范 1. 作用:释放通过malloc/calloc/realloc分配的内存 2. 函数原型:`void free(void* ptr);` 3. 注意: - 只能释放动态分配的内存 - 不能重复释放同一块内存 - 释放后应将指针置为NULL(避免悬空指针) 示例: ```c free(arr); // 释放内存 arr = NULL; // 安全措施 ``` 四、完整使用示例 ```c #include <stdio.h> #include <stdlib.h> int main() { int* numbers; int n, sum = 0; printf("输入元素个数:"); scanf("%d", &n); // 动态分配内存 numbers = (int*)malloc(n * sizeof(int)); if (numbers == NULL) { printf("内存分配失败!"); return 1; } // 使用内存 for (int i = 0; i < n; i++) { scanf("%d", &numbers[i]); sum += numbers[i]; } printf("总:%d\n", sum); // 释放内存 free(numbers); numbers = NULL; return 0; } ``` 五、常见错误 1. 忘记检查malloc返回值 2. 内存泄漏(分配后未释放) 3. 越界访问分配的内存 4. 使用已释放的内存(悬空指针) 5. 错误计算内存大小(如忘记乘sizeof) 六、最佳实践 1. 每个malloc必须对应一个free 2. 释放后立即置指针为NULL 3. 使用calloc初始化内存为零(需要时) 4. 使用realloc调整内存大小(需要时) 5. 推荐使用内存检测工具(如Valgrind) 注意:动态内存管理是C语言的核心特性之一,正确使用可以避免90%以上的内存相关问题。对于长期运行的程序,内存泄漏可能导致严重性能问题!
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值