c语言学习第三十二课---内存开辟位置与柔性数组

简介: c语言学习第三十二课---内存开辟位置与柔性数组

c/c++程序的内存开辟

1.栈区:在执行函数时,函数的内部的储存单元都可以在栈上创建,结束时自动被释放。栈区的分配预案算内置预处理器模块,效率很高,但是分配的内存容量有限。存放不下就会产生栈溢出的现象。

栈区主要存放运行函数时被分配的局部变量,函数参数,返回数据,返回地址等        

2.堆区:一般由程序员分配释放,若若程序员不释放,结束时可能被OS回收,分配方式类似链表。

3.数据段:也是静态区,存放全局变量,静态数据。程序结束时有系统释放。

4.代码段:存放函数体(类成员函数和全局函数)的二进制代码。

526a10656b314fb2b0e5919d5d5f14da.png

realloc函数补充

505c39e138864436a3540ff132cab159.png

realloc也可以像malloc一样申请空间

若第一个参数给的不是所扩容的指针,而是空指针,如realloc(NULL,20),那就是如malloc一样申请20个字节的空间。

柔性数组

在 c99中,结构中的最后一个元素允许是未知大小的数组,这就叫做柔性数组成员。

typedef struct st_type
{
  int i;
  int a[0];//柔性数组成员
};

柔性数组特点1:

.结构体中柔性数组成员前面必须至少有一个其他成员

   两种写法

typedef struct st_type
{
  int i;
  int a[0];//柔性数组成员
};
//或者这样写
struct S
{
  int n;
  char c;
  int arr[];//大小未知
};

柔性数组特点2:

.sizof返回的这种结构大小不包过柔性数组的内存

struct S
{
  int n;
  char c;
  int arr[];//未知,不知道它该占多大空间
};
int main()
{
  printf("%d", sizeof(struct S));//结果为8
}

这里的空间大小不会包括柔性数组成员的大小,大小计算参考之前结构体内容。

柔性数组特点3:

.包含柔性数组成员的结构用malloc函数惊醒内存分配,并且分配的内存应该大于结构体的大小,以适应柔性数组的大小。

在创建结构体变量时,这里创建结构体的方式发生了改变,用动态内存开辟它的空间。

struct S
{
  int n;
  char c;
  int arr[];
};
int main()
{
  //假设柔性数组里我们想放10个整型数据
  struct S*  ptr=(struct S*)malloc(sizeof(struct S)+10*sizeof(int));
  if (ptr== NULL)
  {
    printf("%s\n", strerror(errno));
    return 1;
  }
  //使用
  ptr->n = 100;
  ptr->c = "w";
  for (int i = 0; i < 10; i++)
  {
    ptr->arr[i] = i;
    printf("%d", ptr->arr[i]);
  }
  //调整数组大小  大小调整为20ge 整形
  struct S* tmp = (struct S*)realloc(ptr, sizeof(struct S) + sizeof(int) * 20);
  if (tmp == NULL)
  {
    printf("%s\n", strerror(errno));
    return 1;
  }
  //使用
  ptr->n = 100;
  ptr->c = "w";
  for (int i = 0; i < 20; i++)
  {
    ptr->arr[i] = i;
    printf("%d", ptr->arr[i]);
  }
  //释放
  free(ptr);
  ptr - NULL;
}

利用malloc函数开辟结构体空间大小。

这里的柔性数组成员类似 int arr[10]存放10个整形数据,但是因为该数组的空间是由malloc开辟的,因此我们可是使用realloc函数来对空间进行调整根据我们的需求

通过调整结构体大小动态调整数组大小,柔性可变,故为柔性数组。

通过指针实现柔性数组的动态特点。

这里开辟空间时,只能先开辟结构体的空间,再开辟结构体里指针的空间。

这里的两个例子作用都是存放一到十的10个整形数据,扩容后存放一到二十的20个整型数组。

struct S
{
  int n;
  char c;
  int* arr;
};
int main()
{
  struct S* ptr = (struct S*)malloc(sizeof(struct S));
  if (ptr == NULL)
  {
    printf("%s\n", strerror(errno));
    return 1;
  }
  int* tmp = (int*)malloc(sizeof(int) * 10);
  if (ptr == NULL)
  {
    printf("%s\n", strerror(errno));
    return 1;
  }
  else
  {
    ptr->arr = tmp;//把开辟出来的空间给结构体里的指针
  }
  //使用
  ptr->n = 100;
  ptr->c = "w";
  for (int i = 0; i < 10; i++)
  {
    ptr->arr[i] = i;
    printf("%d", ptr->arr[i]);
  }
  //调整
  int *pc=realloc(ptr->arr, sizeof(int) * 20);
  if (pc == NULL)
  {
    return 1;
  }
  else
  {
    ptr->arr = pc;
  }
  //在使用
  ptr->n = 100;
  ptr->c = "w";
  for (int i = 0; i < 20; i++)
  {
    ptr->arr[i] = i;
    printf("%d", ptr->arr[i]);
  }
  //释放
  free(ptr->arr);
  ptr->arr = NULL;
  free(ptr);
  ptr = NULL;
  return 0;
}

注意:这里再释放时,我们刚开始下开辟的是结构体的空间,后开辟的结构体中指针的空间(扩容也是对结构体里的指针),故释放先释放结构体中指针的空间,在释放结构体的空间。

两者的区别

方案一(柔性数组)

malloc一次,free一次  

容易维护空间,不易出错,且malloc次数少,内存碎片少,空间利用率相对较高。

方案二(指针表示)

malloc两次,free两次,维护难度加大,容易出错  

malooc越多,内存碎片会增多,内存的使用率较低。

其实这里的内存节约也没多少,主要是理解其中特点。


相关文章
|
21天前
|
存储 编译器 程序员
【C语言】内存布局大揭秘 ! -《堆、栈和你从未听说过的内存角落》
在C语言中,内存布局是程序运行时非常重要的概念。内存布局直接影响程序的性能、稳定性和安全性。理解C程序的内存布局,有助于编写更高效和可靠的代码。本文将详细介绍C程序的内存布局,包括代码段、数据段、堆、栈等部分,并提供相关的示例和应用。
32 5
【C语言】内存布局大揭秘 ! -《堆、栈和你从未听说过的内存角落》
|
24天前
|
传感器 人工智能 物联网
C 语言在计算机科学中尤其在硬件交互方面占据重要地位。本文探讨了 C 语言与硬件交互的主要方法,包括直接访问硬件寄存器、中断处理、I/O 端口操作、内存映射 I/O 和设备驱动程序开发
C 语言在计算机科学中尤其在硬件交互方面占据重要地位。本文探讨了 C 语言与硬件交互的主要方法,包括直接访问硬件寄存器、中断处理、I/O 端口操作、内存映射 I/O 和设备驱动程序开发,以及面临的挑战和未来趋势,旨在帮助读者深入了解并掌握这些关键技术。
42 6
|
1月前
|
存储 C语言
C语言如何使用结构体和指针来操作动态分配的内存
在C语言中,通过定义结构体并使用指向该结构体的指针,可以对动态分配的内存进行操作。首先利用 `malloc` 或 `calloc` 分配内存,然后通过指针访问和修改结构体成员,最后用 `free` 释放内存,实现资源的有效管理。
105 13
|
25天前
|
存储 算法 程序员
C 语言指针详解 —— 内存操控的魔法棒
《C 语言指针详解》深入浅出地讲解了指针的概念、使用方法及其在内存操作中的重要作用,被誉为程序员手中的“内存操控魔法棒”。本书适合C语言初学者及希望深化理解指针机制的开发者阅读。
|
1月前
|
存储 C语言 开发者
C 语言指针与内存管理
C语言中的指针与内存管理是编程的核心概念。指针用于存储变量的内存地址,实现数据的间接访问和操作;内存管理涉及动态分配(如malloc、free函数)和释放内存,确保程序高效运行并避免内存泄漏。掌握这两者对于编写高质量的C语言程序至关重要。
52 11
|
23天前
|
并行计算 算法 测试技术
C语言因高效灵活被广泛应用于软件开发。本文探讨了优化C语言程序性能的策略,涵盖算法优化、代码结构优化、内存管理优化、编译器优化、数据结构优化、并行计算优化及性能测试与分析七个方面
C语言因高效灵活被广泛应用于软件开发。本文探讨了优化C语言程序性能的策略,涵盖算法优化、代码结构优化、内存管理优化、编译器优化、数据结构优化、并行计算优化及性能测试与分析七个方面,旨在通过综合策略提升程序性能,满足实际需求。
54 1
|
28天前
|
存储 C语言 计算机视觉
在C语言中指针数组和数组指针在动态内存分配中的应用
在C语言中,指针数组和数组指针均可用于动态内存分配。指针数组是数组的每个元素都是指针,可用于指向多个动态分配的内存块;数组指针则指向一个数组,可动态分配和管理大型数据结构。两者结合使用,灵活高效地管理内存。
|
28天前
|
缓存 Prometheus 监控
Elasticsearch集群JVM调优设置合适的堆内存大小
Elasticsearch集群JVM调优设置合适的堆内存大小
232 1
|
18天前
|
存储 监控 算法
深入探索Java虚拟机(JVM)的内存管理机制
本文旨在为读者提供对Java虚拟机(JVM)内存管理机制的深入理解。通过详细解析JVM的内存结构、垃圾回收算法以及性能优化策略,本文不仅揭示了Java程序高效运行背后的原理,还为开发者提供了优化应用程序性能的实用技巧。不同于常规摘要仅概述文章大意,本文摘要将简要介绍JVM内存管理的关键点,为读者提供一个清晰的学习路线图。
|
27天前
|
Java
JVM内存参数
-Xmx[]:堆空间最大内存 -Xms[]:堆空间最小内存,一般设置成跟堆空间最大内存一样的 -Xmn[]:新生代的最大内存 -xx[use 垃圾回收器名称]:指定垃圾回收器 -xss:设置单个线程栈大小 一般设堆空间为最大可用物理地址的百分之80