`
king_c
  • 浏览: 223806 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

c语言内存分配 2

 
阅读更多

 

二、栈(stack)和堆(heap)具体的区别。

  1、在申请方式上

  栈(stack它由编译器自动管理,无需我们手工控制。 例如,声明函数中的一个局部变量 int b 系统自动在栈中为b开辟空间;在调用一个函数时,系统自动的给函数的形参变量在栈中开辟空间。

  堆(heap申请和释放由程序员控制,并指明大小。容易产生memory leak

  在C中使用malloc函数。

  如:p1 = (char *)malloc(10);

  在C++中用new运算符。

  如:p2 = new char[20];//(char *)malloc(10);

  但是注意p1本身在栈区,而p2本身是在栈中的,只是它们指向的空间是在堆中。

  2、申请后系统的响应上

  栈(stack:只要栈的剩余空间大于所申请空间,系统将为程序提供内存,否则将报异常提示栈溢出。

  堆(heap首先应该知道操作系统有一个记录空闲内存地址的链表,当系统收到程序的申请时, 会遍历该链表,寻找第一个空间大于所申请空间的堆结点,然后将该结点从空闲结点链表中删除,并将该结点的空间分配给程序。另外,对于大多数系统,会在这块内存空间中的首地址处记录本次分配的大小,这样,代码中的deletefree语句才能正确的释放本内存空间。另外,由于找到的堆结点的大小不一定正好等于申请的大小,系统会自动的将多余的那部分重新放入空闲链表中。

3、申请大小的限制

  栈(stack:Windows`,栈是向低地址扩展的数据结构,是一块连续的内存的区域。这句话的意思是栈顶的地址和栈的最大容量是系统预先规定好的,在WINDOWS下,栈的大小是<?xml:namespace prefix = st1 />2M(也有的说是1M,总之是一个编译时就确定的常数),如果申请的空间超过栈的剩余空间时,将提示overflow。因此,能从栈获得的空间较小。 例如,在VC6下面,默认的栈空间大小是1M(好像是,记不清楚了)。

      当然,我们可以修改:打开工程,依次操作菜单如下:Project->Setting->Link,在Category 中选中Output,然后在Reserve中设定堆栈的最大值和commit

  注意:reserve最小值为4Bytecommit是保留在虚拟内存的页文件里面,它设置的较大会使栈开辟较大的值,可能增加内存的开销和启动时间。

  堆(heap堆是向高地址扩展的数据结构,是不连续的内存区域(空闲部分用链表串联起来)。正是由于系统是用链表来存储空闲内存,自然是不连续的,而链表的遍历方向是由低地址向高地址。一般来讲在32位系统下,堆内存可以达到4G的空间,从这个角度来看堆内存几乎是没有什么限制的。由此可见,堆获得的空间比较灵活,也比较大。

  4、分配空间的效率上

  栈(stack:栈是机器系统提供的数据结构,计算机会在底层对栈提供支持:分配专门的寄存器存放栈的地址,压栈出栈都有专门的指令执行,这就决定了栈的效率比较高。但程序员无法对其进行控制。

  堆(heap:C/C++函数库提供的,由newmalloc分配的内存,一般速度比较慢,而且容易产生内存碎片。它的机制是很复杂的,例如为了分配一块内存,库函数会按照一定的算法(具体的算法可以参考数据结构/操作系统)在堆内存中搜索可用的足够大小的空间,如果没有足够大小的空间(可能是由于内存碎片太多),就有可能调用系统功能去增加程序数据段的内存空间,这样就有机会分到足够大小的内存,然后进行返回。这样可能引发用户态和核心态的切换,内存的申请,代价变得更加昂贵。显然,堆的效率比栈要低得多。

  5、堆和栈中的存储内容

  栈(stack:在函数调用时,第一个进栈的是主函数中子函数调用后的下一条指令(子函数调用语句的下一条可执行语句)的地址,然后是子函数的各个形参。在大多数的C编译器中,参数是由右往左入栈的,然后是子函数中的局部变量。注意:静态变量是不入栈的。 当本次函数调用结束后,局部变量先出栈,然后是参数,最后栈顶指针指向最开始存的地址,也就是主函数中子函数调用完成的下一条指令,程序由该点继续运行。

  堆(heap:一般是在堆的头部用一个字节存放堆的大小,堆中的具体内容有程序员安排。

  6、存取效率的比较

  这个应该是显而易见的。拿栈上的数组和堆上的数组来说:

  void main()

  {

  int arr[5]={1,2,3,4,5};

  int *arr1;

  arr1=new int[5];

  for (int j=0;j<=4;j++)

  {

  arr1[j]=j+6;

  }

  int a=arr[1];

  int b=arr1[1];

  }

  上面代码中,arr1(局部变量)是在栈中,但是指向的空间确在堆上,两者的存取效率,当然是arr高。因为arr[1]可以直接访问,但是访问arr1[1],首先要访问数组的起始地址arr1,然后才能访问到arr1[1]

  总而言之:

       C/C++程序中,有关内存使用的问题是最难发现和解决的。这些问题可能导致程序莫名其妙地停 止、崩溃,或者不断消耗内存直至资源耗尽。由于C/C++语言本身的特质和历史原因,程序员使用内存需要注意的事项较多,而且语言本身也不提供类似Java的垃圾清理机制。编程人员使用一定的工具来查找和调试内存相关问题是十分必要的。

       总的说来,与内存有关的问题可以分成两类:内存访问错误和内存使用错误。内存访问错误包括错误地读取内存和错误地写内存。错误地读取内存可能让你的模块返回意想不到的结果,从而导致后续的模块运行异常。错误地写内存可能导致系统崩溃。内存使用方面的错误主要是指申请的内存没有正确释放,从而使程序运行逐渐减慢,直至停止。这方面的错误由于表现比较慢很难被人工察觉。程序也许运行了很久才会耗净资源,发生问题。

  堆和栈的区别可以用如下的比喻来看出:

  使用栈就象我们去饭馆里吃饭,只管点菜(声明变量)、付钱、和吃(使用),吃饱了就走,不必理会切菜、洗菜等准备工作和洗碗、刷锅等扫尾工作,他的好处是快捷,但是自由度小。

  使用堆就象是自己动手做喜欢吃的菜肴,比较麻烦,但是比较符合自己的口味,而且自由度大。

      char s1[] = "hello";

     char s2[] = "hello";

    cout<<(s1==s2)<<endl;   //这里是0,因为s1s2是数组形式的字符,系统会自动在栈区为其分配空间,所以s1s2是指向不同的区域的

    char *s1 = "hello";

     char *s2 = "hello";

     cout<<(s1==s2)<<endl; //这里是1,因为s1s2是指针的形式的字符指针,系统为其分配的空间是在栈区的,但是"hello"是存放在常量区域的,所以s1s2是指向同一区域的,故比较时,是相等的。

 

 

分享到:
评论

相关推荐

    C语言内存分配问题C语言内存分配问题.docx

    C语言内存分配问题详解 在C语言中,内存分配是一个非常重要的概念。正确地分配和释放内存对于程序的正确性和性能至关重要。本文将详细介绍C语言中的内存分配问题,包括内存的四个区、动态分配和释放内存、避免内存...

    c语言内存分配

    ### C语言内存分配详解 #### 一、C语言的4G虚拟寻址空间概念 在C语言中,每个进程都有一个4GB的虚拟地址空间。这个空间被划分为不同的区域来存储不同类型的变量和数据。这里提到的4G指的是4GB(即4 * 1024 * 1024 ...

    c语言中动态内存分配问题

    在C语言中,动态内存分配是一项重要的编程技巧,它允许程序员在程序运行时根据需要分配内存,而不是在编译时预设固定的内存空间。本文将深入解析动态内存分配的细节和一个简单的C语言程序示例。 C语言提供了四个与...

    操作系统 内存分配与回收 C语言模拟(含源代码和.exe文件)

    2. **动态内存分配**:在运行时,通过C语言的`malloc`函数,程序可以请求任意大小的内存块。`malloc`会返回一个指向新分配内存的指针。当不再需要这些内存时,应使用`free`函数将其释放,以防止内存泄漏。 3. **...

    C语言内存精讲.rar

    本资源“C语言内存精讲.rar”包含了深入讲解C语言内存管理的资料,这对于想要深入学习C语言的程序员来说是一份宝贵的参考资料。 C语言的内存主要分为三个区域:栈(Stack)、堆(Heap)和静态存储区(Static Memory...

    C语言内存分配代码.zip

    在C语言中,内存管理是程序员必须关注的重要部分。它涉及到如何为程序的变量和数据结构分配内存,...通过学习和实践这些C语言内存分配的代码,你可以更好地理解和控制程序对内存的使用,从而编写出更健壮、高效的程序。

    C语言资源分配问题代码

    在C语言中,我们可以利用结构体、指针和动态内存分配等特性来模拟这些抽象概念。 1. 结构体:在C语言中,结构体可以用来封装资源的属性,例如资源的类型、数量、当前状态等。通过定义一个资源结构体,我们可以方便...

    C语言的内存分配详解

    在C语言中,内存分配是程序管理资源的关键部分。它涉及到如何在运行时为变量和对象分配内存,以及何时和如何释放这些内存。本篇文章将深入探讨C语言的内存分配,特别是动态内存分配,以及与之相关的堆内存分配。 ...

    c语言内存管理.pdf

    ### C语言内存管理知识点 #### 一、C语言内存管理概览 在C语言中,内存管理是一项核心技能,尤其对于高效程序设计至关重要。本文档将深入探讨C语言中的内存管理机制,帮助开发者理解不同类型的内存分配以及如何...

    C语言内存分布图(计算机基础知识)

     全局/静态存储区,全局变量和静态变量被分配到同一块内存中,在以前的C语言中,全局变量又分为初始化的和未初始化的,在C++里面没有这个区分了,他们共同占用同一块内存区。  常量存储区,这是一块比较特殊的...

    C语言内存管理

    C语言内存管理是一个程序员必须掌握的重要知识点,因为它直接关系到程序运行的稳定性和效率。内存泄漏是C语言中一个非常典型的问题,指的是程序中分配的堆内存无法再被访问或回收,最终导致可用内存不断减少。内存...

    动态内存分配C语言实现

    本项目专注于C语言中的动态内存分配,这在处理数据结构如链表、树或大型数组等时特别有用。在“动态内存分配C语言实现”中,我们将探讨以下几个关键知识点: 1. **malloc() 函数**:这是C语言中用于动态分配内存的...

    C语言动态分配内存ppt课件.ppt

    在C语言中,动态内存分配是通过malloc函数实现的。malloc函数的原型为void *malloc(size_t n),其中n是要分配的内存的大小,返回值是分配内存的块的首地址。 例如,使用malloc函数分配一个可以容纳10个整型元素的...

    c语言 模拟 内存分配管理 用 位视图

    本实验主要是模拟在分页式管理方式下用位视图来表示内存分配情况,实现主存空间的分配和回收。用一个8*8矩阵表示内存的占用状态,1表示已经占用,0表示未占用要求输入作业名字和所需内存块。

    (麻省理工免费课程)C语言内存管理和C++面向对象编程

    1. **`malloc`函数**:`malloc`是C语言中用于动态内存分配的标准库函数,它返回指向已分配内存块的指针。程序员可以通过`free`函数来释放不再使用的内存,以避免内存泄漏。 2. **`new`操作符**:在C++中,`new`...

    C语言动态分配内存.ppt

    C语言动态内存分配 C语言动态内存分配是指在程序运行时分配内存空间,以满足程序的需求。在某些情况下,程序需要在运行时动态地分配内存,以便存储大量数据或处理不确定大小的数据。例如,在顺序对一批文件进行解析...

    C语言的内存分配.pdf

    2. **C语言内存分配函数** - **`malloc()`**:动态分配指定大小的内存,返回指向该内存的指针。分配失败时返回NULL。 - **`calloc()`**:分配指定数量的元素,每个元素的大小为指定的字节数,并将所有内存初始化为...

    虚拟存储器管理——模拟内存分配与回收实验算法(c语言)

    `malloc.h`、`stdio.h`、`iostream.h` 和 `stdlib.h` 是C语言中常用的头文件,它们提供了必要的内存分配和输入/输出功能。 总的来说,这个实验提供了一个实践平台,让学生掌握虚拟存储器管理的基本概念,包括页面...

    内存分配简单的c语言

    在这个问题中,我们讨论的是一个简单的C语言实现,它模拟了内存分配和释放的过程,特别是使用了`malloc()`和`free()`两个函数。这两个函数在C语言中是标准库函数,分别用于动态地分配和释放内存。 `malloc()`函数...

Global site tag (gtag.js) - Google Analytics