C语言学习(三) -- 动态内存分配

动态内存分配 (memory allocation)

内存的开辟

我们平时是在定义变量的时候自动分配一块连续的空间,但内存一旦开辟,就固定死了
因此我们需要给程序员自己开辟和释放内存空间的权力–动态内存管理

库函数(stdlib.h)

alloc – allocation

malloc

这个函数会向系统申请一块连续可用的空间,并返回一个指向这块空间的指针。

void* malloc (size_t size);
  • 如果开辟成功,返回的指针指向这块空间的起始地址;
  • 如果开辟失败,返回NULL。
  • 返回的指针类型是void * ,需要用到什么类型的指针程序员可以自行转换
  • 如果传入的数据是0, 那么malloc怎么做取决于编译器,没啥意义
  • 在接受返回值之后,要确认是否为NULL哦~

注意:malloc申请的空间实在内存的堆区!!!
局部变量和函数的参数是存放在栈区的
全局变量和静态变量是放在静态区的

free

free函数专门用来释放动态内存

void free(void *ptr)
  • 传入的参数一定是开辟的动态内存的地址
  • 如果传入的参数是NULL指针,则free啥事不干

在释放内存之后,别忘了把指针置NULL哦

calloc

calloc函数和malloc很相似,都是用来申请动态内存的,不过这个还会把申请到的内存初始化为0

void* calloc(size_t num, size_t size);
  • calloc会为num个大小为size的元素开辟一块内存空间,并初始化为0
  • 在接受返回值之后,要确认是否为NULL哦~

realloc (re alloc)

当我们在使用动态内存时,有的时候会发现空间不够用或者空间过大的情况,reclloc函数用来调整已经开辟的动态内存的大小

void* realloc(void *ptr, size_t size);
  • 调整空间失败,返回NULL
  • 调整成功
    • 情况一:在已经开辟好的空间后面,没有足够的空间用来直接扩大
      此时realloc会重新再堆区中找1块空间,把旧的数据copy过去,把旧的空间释放掉,并返回新开辟空间的地址
    • 情况二:后面的空间足够,可以在原空间后直接扩容
      此时realloc会直接扩容,并返回原空间的起始地址
  • 传入空指针时等价于malloc
int *p = (int *)(NULL, num)
  • 在接受返回值之后,要确认是否为NULL哦~

建议使用临时变量接收返回值并判断是否为NULL

柔性数组(flexible array)

柔性数组的大小可以根据需要再进行分配大小,使用结构体的形式实现

  • 柔性数组必须在结构体的最后,且前面至少有一个成员
  • sizeof返回的结构体大小不包含柔性数组
  • 使用malloc进行内存分配的时候,要大于结构体的大小
struct flx_array
{
    int i;
    int arr[0]; //有的编译器不能写0,会报错,把0删掉即可
};

int main()
{
    struct flx_array *p = NULL;
    // 使用如下方式进行内存分配 注意观察格式,这样有助于分配和理解
    struct flx_array *p = (struct flx_array *)malloc(sizeof(struct flx_array) + 10 * sizeof(int));
    // ...
    free(p)
    return 0;
}

这样柔性数组arr就获得了10个整型的空间

还有一种方式可以实现和柔性数组相似的功能,在结构体里面使用指针指向数组,但这样做有些麻烦,还有些缺点

  • 二次malloc
  • 空间不连续
  • 使用极不方便
// 示例
typedef struct st_type
{
    int i;
    int *p_a;
};
int main()
{
    type_a *p = (type_a *)malloc(sizeof(type_a));
    p->i = 100;
    p->p_a = (int *)malloc(p->i*sizeof(int))
    // ...
    free(p->p_a);
    p->p_a = NULL;
    free(p);
    p = NULL;
}

柔性数组优势:

  • 方便分配,只需一次malloc即可
  • 方便释放,只需对结构体的指针释放即可
  • 连续的空间访问速度快

感言

昨天晚上爬了尖峰山,收获一场夜景,很不戳,美中不足的是丢掉一张身份证😆

  • 23
    点赞
  • 17
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在C语言中,可以使用malloc函数来动态分配内存给指针。malloc函数的原型为void* malloc(size_t size),它接受一个size_t类型的参数,表示要分配的字节数。malloc函数返回一个指向分配内存的指针,如果分配失败,则返回空指针(NULL)。 使用malloc函数分配内存的基本步骤如下: 1. 定义一个指针变量,用于接收分配内存的地址。 2. 使用malloc函数分配内存,传入所需的字节数作为参数。需要注意的是,malloc函数返回的是void*类型的指针,通常需要进行强制类型转换。 3. 检查malloc函数的返回值是否为NULL,以确保内存分配成功。 4. 如果分配成功,可以通过指针变量进行操作和访问分配内存空间。 5. 使用完毕后,记得使用free函数释放已分配内存,以避免内存泄漏。 以下是一个示例代码,演示如何使用malloc函数分配内存给指针: ```c #include <stdio.h> #include <stdlib.h> int main() { int* ptr = NULL; int n; printf("请输入要分配的元素个数:"); scanf("%d", &n); ptr = (int*) malloc(sizeof(int) * n); if (ptr != NULL) { printf("内存分配成功!\n"); // 可以对ptr指针进行操作和访问分配内存空间 } else { printf("内存分配失败!\n"); } free(ptr); // 释放已分配内存 return 0;} ``` 在上述代码中,我们首先定义了一个指向int类型的指针变量ptr,并初始化为NULL。然后,通过scanf函数获取要分配的元素个数n。接下来,使用malloc函数分配了n个int类型的内存空间,并将返回的地址强制类型转换为int*类型,并赋值给ptr。然后,我们检查ptr是否为NULL,以确定内存分配是否成功。最后,使用free函数释放已分配内存。 需要注意的是,在使用malloc函数分配内存后,需要在不再使用分配内存时及时使用free函数释放内存,以避免内存泄漏问题。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* *3* [C语言学笔记(指针 地址 动态内存分配)](https://blog.csdn.net/m0_62510826/article/details/126639863)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 100%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值