C基础语法(动态内存)

简介: C基础语法(动态内存)

为什么存在动态内存分配

一般我们在存放程序运行需要的数据时,会选择数组来进行存放,但数组的大小是不可变化的,设定了多少就是多少。程序的运行是动态的,我们往往并不知道产生的数据要有多大的空间来存放,如果用数组给的空间太大,会造成内存空间的极大浪费,运行效率也有所降低,给的太少又不够用,导致程序崩溃。因此我们需要一种动态的内存分配机制,如果检测到内存不够用,会自动开辟新的内存空间,用来存放数据。


动态内存分配函数

在C语言中,内存分配函数有3个,分别是 malloc() , calloc(), realloc()

在使用动态内存函数时切记,切记要引头文件 (无数血与泪的教训😭😭)

还有就是不再使用分配的内存后,要记得用free()来释放,否则会造成内存泄露的。

后面再解释内存泄漏,接下来介绍这3个内存开辟函数


malloc

void* malloc (size_t size)

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

如果开辟内存失败则返回空指针,因此在开辟空间时需要对返回的指针进行检查

//用malloc开辟10个int整型空间
int main()
{
  int* p = (int*)malloc(sizeof(int) * 10);
  if (NULL == p)
  {
    printf("fail\n");
    return 1;
  }
  free(p);
  p = NULL;
  return 0;
}
//正常的开辟流程

calloc

void* calloc (size_t num, size_t size);

calloc与malloc在功能上的差别就是calloc能够将开辟的空间都初始化为0,如果需要将开辟的空间初始化,可以选择calloc,另外在参数上也略有区别,calloc要传两个参数,一个参数是要开辟某类型空间的个数,另一个是要开辟的空间的类型。实际上就是把malloc的参数拆开来,上面的malloc( sizeof(int) *10 ) 变成calloc为 calloc( 10, sizeof(int) )

realloc

前面两个函数可能会让你产生一些疑惑,感觉和使用数组开辟的空间区别不大,并没有体现出动态开辟,那么realloc就能够解决这些问题,我们先看看realloc的声明

void* realloc (void* ptr,  size_t size)

由声明可知,第一个参数是一个指针,第二个参数是一个无符号整型数字,分别是什么含义呢。第一个参数需要传指向我们之前开辟过的空间的指针,在之前的例子中,我们尝试用malloc开辟了10个整型空间,并用指针变量p来维护这块空间,假设这10个整型空间不够使用了,我还想再扩增10个整型空间,且保存之前产生的数据,这个时候realloc函数就起到了很大作用。我们把之前维护开辟空间的指针p传过去,然后第二个参数传重新开辟空间的大小,我们想扩增10个容量,加上之前的就是20个空间,然后realloc会返回一个指针,返回的指针分为三种情况

1.空间开辟失败,返回一个空指针

2.原先空间的后面没有被使用的空间能够满足扩增的需求,那就返回维护原先空间的指针

3.原先空间的后面没有被使用的空间不满足扩增的需求,寻找一块新空间,返回新空间的起始地址

接下来我们详细分析一下

int main()
{
  int* p = (int*)malloc(sizeof(int) * 10);
  if (NULL == p)
  {
    printf("fail\n");
    return 1;
  }
// ......
   ......
//经过一段程序的使用,发现原先开辟的空间不够使用,要再扩增10个
 int *ps = (int*)realloc( p, sizeof(int)*20 )
 if( NULL == ps)
   {
    printf("fail\n");
    return 1;
   }
   p = ps; //如果还想让之前的指针p来维护这块新空间,可以进行这一步操作
   ps = NULL;
 //.......
  free(p);
  p = NULL;
  return 0;
}


造成内存泄漏的原因

很多程序一旦运行起来,除了设备停机,基本就一直处于运行状态,如果我们malloc了一块很大的空间,并且在使用完后没有用free()来释放,因为c/c++是没有自动回收机制的,这就导致我们malloc的这块空间在内存中一直处于被占用的状态,不用之后,程序不会再管维护这块空间的指针,就很可能导致维护空间的指针的丢失,从而失去了对这块空间的控制,在程序运行期间,再也找不到这块空间,也没法再使用这块空间,即造成内存泄漏。

而我们平时写的小程序,程序运行一会后就关闭了,此时程序所占用的全部空间归还操作系统,但我们要养成随用随释放的好习惯,不然以后接触大项目,内存泄漏很危险。

造成内存碎片化的原因

我们在用动态内存函数开辟空间时,使用的是内存的堆区,如果我们频繁的使用malloc

realloc来开辟内存空间,会把堆区分成一块一块的,每开辟的两块空间可能就夹杂着一些没用过的空间,而这些空间很细碎很难再被使用,会导致内存的利用率降低。


柔性数组

柔性数组的定义和特点

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

特点:1.结构中的柔性数组成员前面必须至少一个其他成员。

          2.sizeof 返回的这种结构大小不包括柔性数组的内存。

          3.包含柔性数组成员的结构用malloc ()函数进行内存的动态分配,并且分配的内存应                  该大于结构的大小,以适应柔性数组的预期大小

typedef struct stu
{
  int age;
  char name[];
}S;
//char name[]就是一个柔性数组,它的大小可以是未定义的

柔性数组的使用和优势

//柔性数组的使用
struct S
{
    int n;
    int arr[];
};
int main()
{
    structu S s;
    //这样创建会导致柔性数组没有空间
    struct S* ps= (struct S*)malloc(sizeof(struct S) + 40);
    //40是给柔性数组arr预留的大小
}
struct S
{
    int n;
    int *arr;
};
int main() 
{
    struct S* ps = (struct S*)malloc( sizeof(struct S) );
    ps->arr = (int*)malloc(40);
}
//此为普通结构体的使用

柔性数组相对于普通结构体的优势

1.相比较与柔性数组,上面普通结构体会多消耗内存,也要开辟和释放两次

2.因为普通结构体要malloc两次,会造成内存碎片,导致内存利用率降低,运行速度略降低


使用动态内存函数常见的错误

1.对NULL指针进行解引用

造成这种情况的原因主要是不对内存开辟函数返回的指针进行检查,如果不检查,而恰好开辟失败,就造成对空指针的引用

void test()
{
   int *p = (int *)malloc(INT_MAX/4);
   *p = 20;//如果p的值是NULL,就会有问题
   free(p);
}

2.对开辟的空间越界访问

这种情况就是对空间的使用不当,只开辟了10块空间,错误访问到第十一块空间

3.对非动态开辟的内存空间使用free()来释放空间

free()只能用来释放动态开辟的空间

void test()
{
  int a =0;
  int *p = &a;
  free(p);
}
//这样是错误的

4.对同一块动态内存多次释放

一块动态开辟内存释放一次就好,释放多次是错误的

5.使用free释放时只释放了其中一部分

很多时候我们动态开辟了一块空间,用指针p来维护,p指向这块空间的起始地址,但是在使用这块空间时,可能会导致p不再指向起始位置,在不把p返回到起始位置时就把p释放,这样是错误的。

void test()
{
   int *p = (int *)malloc(100);
   p++;
   free(p);//p不再指向动态内存的起始位置
}

6.忘记释放已开辟的内存(内存泄漏)

可能很多同学说自己早已经记住了要释放已开辟的内存,不会造成内存泄露的,事实上我们的逻辑并没有我们认为的那么严谨,不妨看看下面一段程序

int test()
{
  int* p = (int*)malloc(sizeof(int) * 2);
  if (NULL == p)
  {
    printf("fail\n");
    return -1;
  }
  scanf("%d %d", p, p + 1);
  if ( *p > *(p+1) )
  {
    printf("较大的数是:");
    return *p;
  }
  else
  {
    printf("较大的数是:");
    return *(p+1);
  }
  free(p);
  p = NULL;
}
int main()
{
  int m = test();
  printf("%d", m);
  return 0;
}

上面是实现返回较大值的简单程序,如果你感觉这段程序没有毛病的话,那么非常抱歉,你已经造成了内存泄漏,先分析一下原因,我们在test函数内开辟了两个int 类型的空间,并输入两个值进行对比,到目前为止都没有毛病,毛病就出在我们在进行对比后返回一个返回值,直接就离开了test()函数,在离开之前我们可没有对p进行free释放啊,而p又是在test内部定义的,其只能在test内部使用,离开test,p就还给了内存,变成了野指针,也就是说我们再也无法通过p来找到这块未释放的空间,在程序运行期间,这块空间就被泄露了

是不是感觉内存泄漏没有那么简单,如果你一眼就看出了问题,那么你的洞察和逻辑能力很强,很聪明,要继续保持这份警觉,如果你没有看出,那么现在你已经被种上了这份警觉

以后多留意观察,不可盲目自信。


目录
相关文章
|
编译器 C语言 Swift
05-📝C++核心语法|面向对象3【 继承和派生、多态、静态成员、const成员、引用类型成员、VS的内存窗口】
复习`C++核心语法`,且适当进行汇编探索底层实现原理,进一步夯实基础,为以后的`底层开发`、`音视频开发`、`跨平台开发`、`算法`等方向的进一步学习埋下伏笔。
05-📝C++核心语法|面向对象3【 继承和派生、多态、静态成员、const成员、引用类型成员、VS的内存窗口】
|
编译器 C语言 C++
C++基础语法(内存管理)
C++基础语法(内存管理)
82 0
【OC语法快览】四、基础内存管理
Basic Memory Management                                                           基础内存管理 If you're writing an application for Mac OS X, you have the option to enable garbage collection.
774 0
|
2月前
|
缓存 Prometheus 监控
Elasticsearch集群JVM调优设置合适的堆内存大小
Elasticsearch集群JVM调优设置合适的堆内存大小
339 1
|
28天前
|
存储 监控 算法
深入探索Java虚拟机(JVM)的内存管理机制
本文旨在为读者提供对Java虚拟机(JVM)内存管理机制的深入理解。通过详细解析JVM的内存结构、垃圾回收算法以及性能优化策略,本文不仅揭示了Java程序高效运行背后的原理,还为开发者提供了优化应用程序性能的实用技巧。不同于常规摘要仅概述文章大意,本文摘要将简要介绍JVM内存管理的关键点,为读者提供一个清晰的学习路线图。
|
2月前
|
Java
JVM内存参数
-Xmx[]:堆空间最大内存 -Xms[]:堆空间最小内存,一般设置成跟堆空间最大内存一样的 -Xmn[]:新生代的最大内存 -xx[use 垃圾回收器名称]:指定垃圾回收器 -xss:设置单个线程栈大小 一般设堆空间为最大可用物理地址的百分之80
|
2月前
|
Java
JVM运行时数据区(内存结构)
1)虚拟机栈:每次调用方法都会在虚拟机栈中产生一个栈帧,每个栈帧中都有方法的参数、局部变量、方法出口等信息,方法执行完毕后释放栈帧 (2)本地方法栈:为native修饰的本地方法提供的空间,在HotSpot中与虚拟机合二为一 (3)程序计数器:保存指令执行的地址,方便线程切回后能继续执行代码
25 3
|
2月前
|
存储 缓存 监控
Elasticsearch集群JVM调优堆外内存
Elasticsearch集群JVM调优堆外内存
54 1
|
2月前
|
Arthas 监控 Java
JVM进阶调优系列(9)大厂面试官:内存溢出几种?能否现场演示一下?| 面试就那点事
本文介绍了JVM内存溢出(OOM)的四种类型:堆内存、栈内存、元数据区和直接内存溢出。每种类型通过示例代码演示了如何触发OOM,并分析了其原因。文章还提供了如何使用JVM命令工具(如jmap、jhat、GCeasy、Arthas等)分析和定位内存溢出问题的方法。最后,强调了合理设置JVM参数和及时回收内存的重要性。

热门文章

最新文章