`
fireDragonpzy
  • 浏览: 466669 次
  • 性别: Icon_minigender_1
  • 来自: 济南
社区版块
存档分类
最新评论

C++内存基础(一)c语言堆栈问题

阅读更多

 C语言程序编译的内存分配:

  1.栈区(stack) --编译器自动分配释放,主要存放函数的参数值,局部变量值等;

  2.堆区(heap) --由程序员分配释放;

  3.全局区或静态区 --存放全局变量和静态变量;程序结束时由系统释放,分为全局初始化区和全局未初始化区;

  4.字符常量区 --常量字符串放与此,程序结束时由系统释放;

  5.程序代码区--存放函数体的二进制代码

  例: //main.c

  int a=0; //全局初始化区

  char *p1; //全局未初始化区

  void main()

  {

  int b; //栈

  char s[]="bb"; //栈

  char *p2; //栈

  char *p3="123"; //其中,“123\0”常量区,p3在栈区

  static int c=0; //全局区

  p1=(char*)malloc(10); //10个字节区域在堆区

  strcpy(p1,"123"); //"123\0"在常量区,编译器 可能 会优化为和p3的指向同一块区域

  }

  一个C程序占用的内存可分为以下几类:

  (一) 栈

  这是由编译器自动分配和释放的区域。主要存储函数的参数,函数的局部变量等。当一个函数开始执行时,该函数所需的实参,局部变量就推入栈中,该函数执行完毕后,之前进入栈中的参数和变量等也都出栈被释放掉。它的运行方式类似于数据结构中的栈。

  (二) 堆

  这是由程序员控制分配和释放的区域,在C里,用malloc()函数分配的空间就存在于堆上。在堆上分配的空间不像栈一样在某个函数执行完毕就自动释放,而是一直存在于整个程序的运行期间。当然,如果你不手动释放(free()函数)这些空间,在程序运行结束后系统也会将之自动释放。对于小程序来说可能感觉不到影响的存在,但对于大程序,例如一个大型游戏,就会遇到内存不够用的问题了

  (三) 全局区

  C里的全局变量和静态变量存储在全局区。它们有点像堆上的空间,也是持续存在于程序的整个运行期间,但不同的是,他们是由编译器自己控制分配和释放的。

  (四) 文字常量区

  例如char *c = “123456”;则”123456”为文字常量,存放于文字常量区。也由编译器控制分配和释放。

  (五) 程序代码区

  存放函数体的二进制代码。

  2. 例子(一)

  int a = 0; //全局区

  void main()

  {

  int b; //栈

  char s[] = "abc"; //s在栈,"abc"在文字常量区

  char *p1,*p2; //栈

  char *p3 = "123456"; //"123456"在常量区,p3在栈上

  static int c =0; //全局区

  p1 = (char *)malloc(10); //p1在栈,分配的10字节在堆

  p2 = (char *)malloc(20); //p2在栈,分配的20字节在堆

  strcpy(p1, "123456"); //"123456"放在常量区

  //编译器可能将它与p3所指向的"123456"优化成一个地方。

  }

  3. 例子(二)

  //返回char型指针

  char *f()

  {

  //s数组存放于栈上

  char s[4] = {'1','2','3','0'};

  return s; //返回s数组的地址,但程序运行完s数组就被释放了

  }

  void main()

  {

  char *s;

  s = f();

  printf ("%s", s); //打印出来乱码。因为s所指向地址已经没有数据

  }

  还有就是函数调用时会在栈上有一系列的保留现场及传递参数的操作。

  栈的空间大小有限定,vc的缺省是2M。栈不够用的情况一般是程序中分配了大量数组和递归函数层次太深。有一点必须知道,当一个函数调用完返回后它会释放该函数中所有的栈空间。栈是由编译器自动管理的,不用你操心。

  堆是动态分配内存的,并且你可以分配使用很大的内存。但是用不好会产生内存泄漏。并且频繁地malloc和free会产生内存碎片(有点类似磁盘碎片),因为C分配动态内存时是寻找匹配的内存的。而用栈则不会产生碎片,在栈上存取数据比通过指针在堆上存取数据快些。一般大家说的堆栈和栈是一样的,就是栈(stack),而说堆时才是堆heap.栈是先入后出的,一般是由高地址向低地址生长。

  堆(heap)和栈(stack)是C/C++编程不可避免会碰到的两个基本概念。首先,这两个概念都可以在讲数据结构的书中找到,他们都是基本的数据结构,虽然栈更为简单一些。在具体的C/C++编程框架中,这两个概念并不是并行的。对底层机器代码的研究可以揭示,栈是机器系统提供的数据结构,而堆则是C/C++函数库提供的。具体地说,现代计算机(串行执行机制),都直接在代码底层支持栈的数据结构。这体现在,有专门的寄存器指向栈所在的地址,有专门的机器指令完成数据入栈出栈的操作。种机制的特点是效率高,支持的数据有限,一般是整数,指针,浮点数等系统直接支持的数据类型,并不直接支持其他的数据结构。因为栈的这种特点,对栈的使用在程序中是非常频繁的。对子程序的调用就是直接利用栈完成的。机器的call指令里隐含了把返回地址推入栈,然后跳转至子程序地址的操作,而子程序中的ret指令则隐含从堆栈中弹出返回地址并跳转之的操作。C/C++中的自动变量是直接利栈的例子,这也就是为什么当函数返回时,该函数的自动变量自动失效的原因。

  和栈不同,堆的数据结构并不是由系统(无论是机器系统还是操作系统)支持的,而是由函数库提供的。基本的malloc/realloc/free函数维护了一套内部的堆数据结构。当程序使用这些函数去获得新的内存空间时,这套函数首先试图从内部堆中寻找可用的内存空间,如果没有可以使用的内存空间,则试图利用系统调用来动态增加程序数据段的内存大小,新分配得到的空间首先被组织进内部堆中去,然后再以适当的形式返回给调用者。当程序释放分配的内存空间时,这片内存空间被返回内部堆结构中,可能会被适当的处理(比如和其他空闲空间合并成更大的空闲空间),以更适合下一次内存分配申请。这套复杂的分配机制实际上相当于一个内存分配的缓冲池(Cache),使用这套机制有如下若干原因:

  1. 系统调用可能不支持任意大小的内存分配。有些系统的系统调用只支持固定大小及其倍数的内存请求(按页分配);这样的话对于大量的小内存分类来说会造成浪费。

  2. 系统调用申请内存可能是代价昂贵的。系统调用可能涉及用户态和核心态的转换。

  3. 没有管理的内存分配在大量复杂内存的分配释放操作下很容易造成内存碎片

  堆和栈的对比

  从以上知识可知,栈是系统提供的功能,特点是快速高效,缺点是有限制,数据不灵活;而堆是函数库提供的功能,特点是灵活方便,数据适应面广泛,但是效率有一定降低。栈是系统数据结构,对于进程/线程是唯一的;堆是函数库内部数据结构,不一定唯一。不同堆分配的内存无法互相操作。栈空间分静态分配和动态分配两种。静态分配是编译器完成的,比如自动变量(auto)的分配。动态分配由alloca函数完成。栈的动态分配无需释放(是自动的),也就没有释放函数。为可移植的程序起见,栈的动态分配操作是不被鼓励的!堆空间的分配总是动态的,虽然程序结束时所有的数据空间都会被释放回系统,但是精确的申请内存/释放内存匹配是良好程序的基本要素。

  引言:对于指针,正确的分配动态内存是十分重要的,本文将着重阐述动态内存分配函数malloc,calloc,realloc以及memset的用法。

    一、对于malloc,在终端输入 #:man malloc可以知道函数原型是:

    Void *calloc(size_t size) ,包含在库函数 stdlib.h中,作用是在内存的堆区分配一个大小为size的连续空间,如果分配内存成功,函数返回新分配内存的首地址,否则,返回NULL,注意:鉴于上述这点,一般在写程序需要判断分配内存是否成功,如下程序语句:

    int *p;

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

    if(p!=NULL)

    .................................//需要执行的语句

    else

    .........................//打印分配内存不成功出错信息

    通常造成内存分配失败的原因如下:

    1、 内存访问越界

    2、 所需连续的内存空间不足

    二、对于函数calloc用法大致与malloc相同,函数原型为:

  void *callo(size_t num,size_t size),作用是在内存中分配连续大小为num*size的空间,这一点在动态数组内存分配有所体现,返回值以及判断返回是否成功与上面相同,下面重点来讨论malloc与calloc区别:

  (2)calloc函数

  函数原型:void *calloc(unsigned n,unsigned size)

  功能:在内存的动态存储区中分配n个长度为size字节的连续空间。返回值为所分配存储区的起始地址,分配不成功则返回NULL。

  例如:char *p;

  p=(char *)calloc(2,20);

  【说明】表示分配2块大小为20个字节的连续存储空间并进行强制类型转换,p代表首地址。

  【注意】与malloc函数相比,calloc函数可以一次分配n块区域。??????

    1、后者在返回指向内存的指针之前把它初始化为0。

    2、请求内存数量的方式不同。malloc的参数仅仅是需要分配的内存字节数;calloc的参数包括元素的数量和每个元素的字节数。

    为了说明第一点,请看如下程序:

    程序在第6行动态为指针p动态分配了内存, 经过gcc编译,运行结果如下:

  注意:6593第一行数据

    由图可以看出红色标记部分,并没有初始化为零,也就是说在这个单元存在随机数,这样程序在运行时可能会出错。将上面的程序用calloc来调用,程序如下:

    见上述程序第6行,用calloc来代替malloc分配内存单元,运行结果如下:

  注意:全为0

    可以看出在用calloc申请内存时将内存都初始化为0了。那么有没有用办法用malloc同时又将内存初始化为0呢?答案是有的,用memset可以实现这一功能将第一个程序做相应改动,程序如下:

    在第七行添加了语句注意:正确的应该是

  memset(p,0,100*sizeof(int)),这条语句的意思是在内存单元p所指向的100个内存单元都赋值为0,相当与初始化内存。此时在运行此程序将不会再出现形如上述红色标记部分的结果。注意:当是0或-1是,输出结果是0和-1,其他得数输出时和自己给的值不一样,如1时输出16843009????????

    三、对于realloc(),函数原型是*void realloc(void *ptr,size_t size),改变ptr所指内存区域的大小为size长度,如果重新分配成功则返回指向被分配内存的指针,否则返回空指针NULL。当内存不再使用时,应使用free()函数将内存块释放。有一点需要注意:当分配内存成功之后,应将原本的指针ptr=NULL,否则会形成野指针,可能造成系统崩溃。

  提示:不论是以上那种方式申请内存,在申请内存之后,最终都要用free释放空间,不然会造成内存泄漏。

  memest原型 (please type "man memset" in your shell) 

  void *memset(void *s, int c, size_t n); 

  memset:作用是在一段内存块中填充某个给定的值,它对较大的结构体或数组进行清零操作的一种最快方法。

  常见的三种错误

  第一: 搞反了c 和 n的位置. 

  一定要记住 如果要把一个char a[20]清零, 一定是 memset(a, 0, 20) 

  而不是 memset(a, 20, 0) 

  第二: 过度使用memset, 我想这些程序员可能有某种心理阴影, 他们惧怕未经初始化的内存, 所以他们会写出这样的代码: 

  char buffer[20]; 

  memset(buffer, 0, sizeof((char)*20)); 

  strcpy(buffer, "123"); 

  这里的memset是多余的. 因为这块内存马上就被覆盖了, 清零没有意义. 

  第三: 其实这个错误严格来讲不能算用错memset, 但是它经常在使用memset的场合出现 

  int some_func(struct something *a){ 

  … 

  … 

  memset(a, 0, sizeof(a)); 

  … 

  } 

  问:为何要用memset置零?memset( &Address, 0, sizeof(Address));经常看到这样的用法,其实不用的话,分配数据的时候,剩余的空间也会置零的。

  答:1.如果不清空,可能会在测试当中出现野值。 你做下面的试验看看结果() 

  char buf[5]; 

  CString str,str1; //memset(buf,0,sizeof(buf)); for(int i = 0;i



摘自:http://blog.gxnews.com.cn/u/130193/a/839674.html


分享到:
评论

相关推荐

    C++写括号匹配堆栈

    本题目以"C++写括号匹配堆栈"为标题,旨在通过实现一个程序来检查输入的字符串或文件中的括号是否正确配对。下面将详细讨论相关知识点。 首先,我们要了解C++语言。C++是一种强大的、面向对象的编程语言,具有高效...

    计算机二级c++之C++与C语言的区别.pdf

    C++是C语言的一个扩展,它引入了许多面向对象编程的特性,使得程序设计更为灵活和高效。C++的发展始于1980年,由Bjarne Stroustrup博士基于C语言构建,旨在融合面向对象的概念。C++在1985年开始流行,并于1998年成为...

    c语言算法堆栈等包括八皇后问题

    c语言的常用算发,和一些常见的程序代码括八皇后问题,约瑟夫环C语言!

    手撕代码_C语言手撕_C++_手撕代码c++_

    总之,“手撕代码_C语言手撕_C++_手撕代码c++_”这个资源旨在帮助面试者准备使用C++解决数据结构和算法问题,特别是通过队列实现堆栈这一技巧,有助于提升面试者的编程思维和问题解决能力。在准备面试时,不仅需要...

    C语言编写devc++

    对于初学者,Dev-Cpp还有一个优点是它包含了标准库头文件,例如`<stdio.h>`、`<stdlib.h>`等,这些都是C语言中的基本输入输出和内存管理函数。熟悉这些函数的使用是编写C程序的重要部分。 总的来说,Dev-Cpp作为...

    ARM汇编 C语言 C++ 相互调用

    这在C++代码中调用C语言编写的库时非常有用,因为它可以解决C++编译器可能产生的名字修饰问题。 - 在C++中调用C库的例子包括创建C动态库,使用gcc编译器的shared选项和lib选项将其编译为共享库,并使用extern "C"来...

    堆栈计算器,支持各级运算,采用边计算边压栈的算法

    在计算机科学中,堆栈常用于实现许多基础功能,如函数调用、表达式求值、内存管理等。在这个项目中,“堆栈计算器”是一个能够处理数学表达式的程序,它利用堆栈来解决复杂的运算问题,包括优先级运算和括号处理。 ...

    C语言学习资料软件

    内存管理也是C语言的一大特色,学习者需要理解堆栈和堆的区别,学会动态内存分配(malloc、calloc、realloc、free)以及如何避免内存泄漏。此外,指针是C语言的灵魂,理解和运用指针可以实现高效的数据操作和复杂的...

    C语言学习路线图:(阶段六) C语言必须知道的300个问题

    C语言是一种基础且强大的...每个问题都可能涉及到一个或多个知识点,通过解答这些问题,你可以系统地提升C语言的综合能力,为后续的软件开发打下坚实基础。高清PDF文档将提供详细的解答和示例,有助于深入学习和理解。

    C++基础教程PPT课件

    C++是由Bjarne Stroustrup在C语言基础上扩展开发的,它保留了C语言的大部分语法,并引入了类、模板、异常处理等面向对象特性。C++不仅支持面向过程编程,还支持面向对象编程,以及泛型编程和函数式编程等编程范式。 ...

    c语言深度解剖&高质量c、c++编程

    《C语言深度解剖》与《高质量C/C++编程》是两本深受程序员喜爱的书籍,它们涵盖了C语言和C++的基础以及高级编程概念。在深入理解这两种编程语言的过程中,学习者可以掌握到一系列关键的知识点。 首先,C语言作为...

    计算机c语言与c++使用工具

    C语言是一种基础且强大的编程语言,被广泛应用于系统编程、嵌入式开发、游戏引擎等多个领域。而C++是C语言的扩展,增加了面向对象编程的概念,使得代码更加模块化和可重用。 描述中提到的“vc6.0绿色完整版”是指...

    windbg c++开发调试工具

    Windbg是一款强大的Windows调试工具,尤其在C++开发过程中,它是不可或缺的利器。这款工具以其丰富的功能和深度的系统级调试能力,深受程序员和系统管理员的喜爱。作为一个绿色免安装版本,它无需复杂的安装过程,只...

    c语言实现简单计算器

    本项目“C语言实现简单计算器”是利用C语言编写的一个基础计算工具,它借助了数据结构中的堆栈来实现计算功能。在本文中,我们将深入探讨堆栈的概念,C语言的语法应用,以及如何在Windows下的Turbo C++(简称win-tc...

    大学C语言程序设计下载

    2. **指针**:C语言的核心特性之一,指针允许直接操作内存地址,可以实现高效的内存管理。理解指针的概念、声明、赋值和解引用是学习C语言的关键。 3. **数组和字符串**:数组是相同类型元素的集合,而字符串是字符...

    浅析ARM汇编 C语言 C++ 相互调用的方法

    本文介绍了ARM汇编、C语言与C++相互调用的基础方法及其应用场景。通过具体实例的讲解,希望能帮助读者更好地理解和掌握不同语言间交互的技术细节。在实际开发过程中,合理运用这些方法能够提高代码的复用性和系统的...

Global site tag (gtag.js) - Google Analytics