嵌入式linux之高级c语言专题—数组&字符串&结构体&共用体&枚举

本文深入探讨了C语言在嵌入式Linux环境中的内存管理,包括栈、堆和数据区的来源及其特点。详细阐述了栈内存的自动分配与回收、堆内存的手动申请与释放,以及数据区的内存分配。同时,讲解了C语言中字符串的本质和结构体、共用体的使用,枚举的概念。通过对内存管理的理解,有助于提高程序设计和优化的水平。
摘要由CSDN通过智能技术生成

数组&字符串&结构体&共用体&枚举


1.1.程序中内存从哪里来1

1.1.1、程序执行需要内存支持

对程序来说,内存就是程序的立足之地(程序是被放在内存中运行的);程序运行时需要内存来存储一些临时变量。

1.1.2、内存管理最终是由操作系统完成的

(1)内存本身在物理上是一个硬件器件,由硬件系统提供。

(2)内存是由操作系统统一管理。为了内存管理方便又合理,操作系统提供了多种机制来让我们应用程序使用内存。这些机制彼此不同,各自有各自的特点,我们程序根据自己的实际情况来选择某种方式获取内存(在操作系统处登记这块内存的临时使用权限)、使用内存释放内存(向操作系统归还这块内存的使用权限)。

1.1.3、三种内存来源:栈(stack)、堆(heap)、数据区(.data

在一个C语言程序中,能够获取的内存就是三种情况:栈(stack)、堆(heap)、数据区(.data

1.1.4、栈的详解

运行时自动分配&自动回收:栈是自动管理的,程序员不需要手工干预。方便简单。

反复使用:栈内存在程序中其实就是那一块空间,程序反复使用这一块空间。

脏内存:栈内存由于反复使用,每次使用后程序不会去清理,因此分配到时保留原来的值。(故局部变量值是随机的)

临时性:(函数不能返回栈变量的指针,因为这个空间是临时的

栈会溢出:因为操作系统事先给定了栈的大小,如果在函数中无穷尽的分配栈内存总能用完。

1.2.程序中内存从哪里来2

1.2.1、堆内存详解

操作系统堆管理器管理:堆管理器是操作系统的一个模块,堆管理内存分配灵活,按需分配

大块内存:堆内存管理者总量很大的操作系统内存块,各进程可以按需申请使用,使用完释放

程序手动申请&释放:手工意思是需要写代码去申请malloc和释放free

脏内存:堆内存也是反复使用的,而且使用者用完释放前不会清除,因此也是脏的。

临时性堆内存只在mallocfree之间属于我这个进程,而可以访问。在malloc之前和free之后都不能再访问,否则会有不可预料的后果。

1.2.2、堆内存使用范例

 void *malloc(size_t size);
 void *calloc(size_t nmemb, size_t size);
 void *realloc(void *ptr, size_t size);

(1)void *是个指针类型,malloc返回的是一个void *类型的指针,实质上malloc返回的是堆管理器分配给我本次申请的那段内存空间的首地址malloc返回的值其实是一个数字,这个数字表示一个内存地址)。

(2)为什么要使用void *作为类型?

主要原因是malloc帮我们分配内存时只是分配了内存空间,至于这段空间将来用来存储什么类型的元素malloc是不关心的,由我们程序自己来决定。

(3)什么是void类型。早期被翻译成空型,这个翻译非常不好,会误导人。void类型不表示没有类型,而表示万能类型void的意思就是说这个数据的类型当前是不确定的,在需要的时候可以再去指定它的具体类型。void *类型是一个指针类型,这个指针本身占4个字节,但是指针指向的类型是不确定的,换句话说这个指针在需要的时候可以被强制转化成其他任何一种确定类型的指针,也就是说这个指针可以指向任何类型的元素。

(4)malloc的返回值:成功申请空间后返回这个内存空间的指针,申请失败时返回NULL。所以malloc获取的内存指针使用前一定要先检验是否为NULL

 

(5)malloc申请的内存时用完后要free释放。free(p);会告诉堆管理器这段内存我用完了你可以回收了。堆管理器回收了这段内存后这段内存当前进程就不应该再使用了。因为释放后堆管理器就可能把这段内存再次分配给别的进程,所以你就不能再使用了。

 void free(void *ptr);

(6)再调用free归还这段内存之前,指向这段内存的指针p一定不能丢(也就是不能给p另外赋值)。因为p一旦丢失这段malloc来的内存就永远的丢失了(内存泄漏),直到当前程序结束时操作系统才会回收这段内存。

#include <stdio.h>

#include <stdlib.h>

int main(void)
{
//第一步:申请堆内存

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

//第二步:检验分配成功

if(NULL ==p) 
{

	printf("malloc error\n");
}

//第三步:使用申请到的内存
*(p+0) = 1;
*(p+1) = 2;
printf("*(p+0) = %d\n",*(p+0));
printf("*(p+1) = %d\n",*(p+1));

//注意:free前不能给p赋值,否则会导致内存泄露

//第四步:释放
free(p);
P = NULL; //防止野指针

*(p+0) = 222;
*(p+1) = 111;

//堆管理器回收了这段内存后这段内存当前进程就不应该再使用了
printf("*(p+0) = %d\n",*(p+0));
printf("*(p+1) = %d\n",*(p+1));

return 0;
}

 

1.2.3malloc的一些细节表现

malloc(0)

malloc申请0字节内存本身就是一件无厘头事情,一般不会碰到这个需要。

如果真的malloc(0)返回的是NULL还是一个有效指针?

答案是:实际分配了16Byte的一段内存并且返回了这段内存的地址。这个答案不是确定的,因为C语言并没有明确规定malloc(0)时的表现,由各malloc函数库的实现者来定义。

malloc(4)

gcc中的malloc默认最小是以16B为分配单位的。如果malloc小于16B的大小时都会返回一个16字节的大小的内存。malloc实现时没有实现任意自己的分配而是允许一些大小的块内存的分配。

malloc(20)去访问第25、第250、第2500····会怎么样

实战中:120字节处正确,1200字节处正确····终于继续往后访问总有一个数字处开始段错误了。

1.3.

  • 1
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值