C语言内存四区的学习总结(二)---- 堆区

接上篇,内存四区的分析-静态区,下面来说明一下堆区总结。

堆区分析:

堆区(heap):一般由程序员分配释放(动态内存申请与释放),若程序员不释放,程序结束时可能由操作系统回

就下面的程序:

#include "stdio.h"
#include "string.h"

char *getMem(int size)
{
    char *p = NULL;
    p = (char *)malloc(sizeof(char) * size);
    if(p == NULL) return NULL;

    return p;
}
int main(int argc, const char **argv)
{
    char *tmp = NULL;
    tmp = getMem(128);
    if(tmp == NULL) return -1;
    strcpy(tmp, "aabbccdd");
    printf("tmp = %s\r\n", tmp);
    free(tmp1);
    return 0;
}

在main开始执行之前,先进行假设栈区的开口向上,栈区中的横向形象的比方函数之间分割线,那么,在main开始执行的时候,

1、先定义了一个指针变量tmp;那么此时在栈区会进栈 tmp,标识为①;

2、开始函数getMem的调用

3、然后再函数getMem中,首先定义一个指针变量p,标识为②

4、然后根据malloc开始申请内存空间,为128字节(白色表示),标识为③

5、然后将malloc返回的地址赋值给p,假设malloc返回的地址为0xaabb;则标识为④

6、因为变量p是在栈区,在函数getMem返回后,p被析构(灰色表示,但是在堆区的内存空间没有被析构),但是在函数getMem返回的是p的地址,所以讲p的地址赋值给tmp,标识为⑤

7、将字符串???????"aabbccdd"拷贝到tmp指向的内存空间,表示为⑥

关于函数strcpy的相关用法,详情查看:http://www.runoob.com/cprogramming/c-function-strcpy.html
        但需要注意的是,strcpy拷贝字符串的时候并不包含结尾的’\0’。

相同功能的还有函数strncpy函数:http://www.runoob.com/cprogramming/c-function-strncpy.html
8、将字符串打印出来

9、释放先前申请的内存(灰色表示),归还给操作系统。

10、函数main返回,执行程序结束。

综上所述,可以得出其内存四区的效果为:

?

编译执行后的效果为:

?

注意:在用malloc申请内存后,如果后续这个内存不在使用,那应该是及时的将这个内存free掉,不然会引起内存泄漏。

内存泄漏也称作"存储渗漏",用动态存储分配函数动态开辟的空间,在使用完毕后未释放,结果导致一直占据该内存单元。直到程序结束。(其实说白了就是该内存空间使用完毕之后未回收)即所谓内存泄漏。

内存泄漏形象的比喻是"操作系统可提供给所有进程的存储空间正在被某个进程榨干",最终结果是程序运行时间越长,占用存储空间越来越多,最终用尽全部存储空间,整个系统崩溃。所以"内存泄漏"是从操作系统的角度来看的。这里的存储空间并不是指物理内存,而是指虚拟内存大小,这个虚拟内存大小取决于磁盘交换区设定的大小。由程序申请的一块内存,如果没有任何一个指针指向它,那么这块内存就泄漏了。

原文地址:https://www.cnblogs.com/songshuaiStudy/p/10653354.html

时间: 2024-08-28 06:38:03

C语言内存四区的学习总结(二)---- 堆区的相关文章

c语言内存四区模型

c语言内存四区模型:代码区,全局区(常量区),栈区,堆区 在全局区(常量区),两个字符串完全一样c++编译器只会定义一份 char * getBuf() { char buf[20]; strcpy(buf, "abcde"); return buf; } abcdX?  有乱码! 确实把内存地址返回了,但不能用 被调函数调用完毕,在临时区分配的内存统统消失 char *buf= (char *)malloc(sizeof(char)*20); 手动malloc申请一份内存,由程序员手

C语言 内存四区与函数调用模型

C语言提高笔记 标签(空格分隔): C++ C语言 day1 数组做函数参数的退回问题 数组做函数参数会退回为一个指针, 正确做法:把数组的内存首地址和数组的有效长度传给被调用函数. 实参的a 和 形参的a 的数据类型本质不一样, 形参中的数组,编译器会把它当成指针处理 只会分配四个字节. 形参写在函数上,和写在函数内是一样的,只不过是具有对外的属性而已. 数据类型本质分析 数据类型可理解为创建变量的模具(模子):是固定内存大小的别名: 数据类型的作用:编译器预算对象(变量)分配的内存空间大小

C语言内存四区

按照老版操作系统来学习,内存对于程序来讲分四区.分别是 代码区,静态区,栈,堆. 由上面程序执行的结果可知: 貌似结果就是 静态代码堆栈 静态区存放的是程序中所有静态变量和常量的值.静态区的大小是程序加载到内存之后就固定的,不会再发生改变. 代码区中存放的是程序中的代码,不能修改它的值,只能通过指针或者变量名来使用函数.[当然强行修改是被允许的但是可能发生未知错误.] 堆是一块很大的区域,通常情况下是4G的大小,(这个数字通常情况下是虚拟的,但是可以利用现代操作系统来利用硬盘模拟出来这么一块儿空

面向过程—面向对象(C++的封装,this)_内存四区_变量生命周期

1.面向对象主要涉及  构造函数.析构函数.虚函数.继承.多态等. 2.对各种支持 的底层实现机制 c语言中,数据 和 处理数据的操作(函数) 是分开来声明,即语言本身并没有支持 “数据和函数”的关联性. 在C++中,通过抽象数据类型(abstract data type, ADT),在类中定义数据和函数,来实现数据和函数直接的绑定. C++成员数据:static.nonstatic C++成员函数:static.nonstatic.virtual C++中的类class从面向对象理论出发,将变

C语言的内存管理分析 栈区 代码区 堆区 静态区 常量区

系统为了管理内存 把内存划分了几个区域 1> 栈区 栈区之中的数据在栈区之中以栈的形式进行存储. 栈区的特点:数据是先进后出, 放在栈区里面存放的是局部变量.(例如定义在函数内部的变量) 栈区之中的数据(局部变量)的作用范围过了之后,系统就会回收自动管理栈区的内存(分配内存 , 回收内存),不需要开发人员来手动管理 2> 堆区 高效的使用内存 这里的内存可以由程序员自己手动管理 高效的使用内存  例如: 申请内存 释放内存.优化内存 ARC 申请内存的函数 malloc()  memary a

iOSDay08之C语言内存管理

本次主要学习和理解C语言中的内存管理 1.存储区划分 按照地址从高到低的顺序:栈区,堆区,静态区,常量区,代码区 1> 栈区:局部变量的存储区域 局部变量基本都在函数.循环.分支中定义 栈区的内存空间由系统自动分配和回收 栈顶,栈底:先定义的局部变量存储区域从栈底开始分配,后定义的局部变量向栈顶分配 特点:先进后出,后进先出 当函数.循环.分支结束后,局部变量的生命周期结束,不能被使用,由系统自动回收内存空间 1 void test1() { 2 int a = 100; 3 } 4 5 voi

C语言内存使用的常见问题及解决之道

一  前言 本文所讨论的“内存”主要指(静态)数据区.堆区和栈区空间(详细的布局和描述参考<Linux虚拟地址空间布局>一文).数据区内存在程序编译时分配,该内存的生存期为程序的整个运行期间,如全局变量和static关键字所声明的静态变量.函数执行时在栈上开辟局部自动变量的储存空间,执行结束时自动释放栈区内存.堆区内存亦称动态内存,由程序在运行时调用malloc/calloc/realloc等库函数申请,并由使用者显式地调用free库函数释放.堆内存比栈内存分配容量更大,生存期由使用者决定,故

C语言提高之技术模型层次、学习标准、特点、内存四区、函数调用模型

1.C语言技术模型分层: 其中,接口的封装和设计尤为重要! 2.着重需要培养的能力 (1)接口的封装和设计(业务模型的抽象.功能抽象和封装) ---重中之重! a.接口api的使用能力: b.接口api的查找能力(快速上手): c.接口api的实现能力: // SOCKETCLIENT_H,一个简单信息系统的封装接口 #ifndef _SOCKETCLIENT_H #endif _SOCKETCLIENT_H #ifdef __cplusplus extern "C" { #endif

【C语言】程序内存四区

1.栈区(临时区)(stack) 由编译器自动分配释放,存放函数的参数值,局部变量的值等: 2.堆区(heap) 一般由程序员分配释放(动态内存申请与释放),若程序员不释放,程序结束时可能有操作系统回收: 3.全局区(静态区)(static) 全局变量和静态变量的存储是放在一起的,初始化的全局变量和静态变量在一块区域,未初始化的全局变量和未初始化的静态变量在相邻的另一块区域,该区域在程序结束后由操作系统释放: 常量区——字符串常量和其他常量的存储位置,程序结束后由操作系统释放: 4.程序代码区