当前位置: 首页 > news >正文

C++系列-内存模型

内存模型

  • 内存模型四个区
  • 代码区
  • 全局区
  • 栈区
  • 堆区
    • 内存开辟和释放
      • 在堆区开辟数组

内存模型四个区

不同区域存放的数据生命周期是不同的,更为灵活。

  • 代码区:存放函数体的二进制代码,操作系统管理。
  • 全局区:存放全局变量,常量,静态变量。
  • 栈区:编译器自动分配释放,存放函数的参数值,局部变量等。
  • 堆区:由程序员分配和释放,如果不人为操作,则程序执行完之后由操作系统回收。

代码区

  • 在程序编译完,生成exe文件,未执行该程序前分为两个区域,代码区和全局区。
  • 代码区存放CPU执行的机器指令。
  • 代码区是共享的,对于频繁执行的程序,打开几次exe文件,执行的是同一块代码区。
  • 代码区的内容是只读的,防止程序意外的更改了指令。

全局区

  • 在程序执行前就存在。
  • 全局区的数据在程序执行完毕后,由操作系统释放。
  • 全局变量,静态变量存放在全局区。
  • 字符串常量和全局常量存放在全局区。
  • 局部常量不在。
code:#include<iostream>using namespace std;int G_a = 66;const int C_G_a = 88;void main(){static int S_a = 88;int a = 10, b = 30;const int C_a = 100;cout << "局部变量a的地址是:" << &a << endl;cout << "局部变量b的地址是:" << &b << endl;cout << "全局变量G_a的地址是:" << &G_a << endl;	cout << "静态变量S_a的地址是:" << &S_a << endl;cout << "字符串常量的地址是:" << &"hello" << endl;cout << "const全局变量C_G_a的地址是:" << &C_G_a << endl;cout << "const局部变量C_a的地址是:" << &C_a << endl;system("pause");}
result:临时变量a的地址是:00000024145FFC54临时变量b的地址是:00000024145FFC74全局变量G_a的地址是:00007FF7011EF050静态变量S_a的地址是:00007FF7011EF054C_a的地址是:00000024145FFCB4字符串常量的地址是:00007FF7011EBCA8str的地址是:00000024145FFC94

栈区

  • 编译器自动分配释放,存放函数的参数值,局部变量等。
  • 在函数中不要返回局部变量的地址。
在函数调用完后,局部变量存放于栈区,会由编译器释放,返回地址的话再引用这个地址可能已经被释放
#include<iostream>
using namespace std;
int G_a = 10;int* test()
{int a = 10;return &a;			// 返回了局部变量的地址,可能会出错
}void main()
{int* p = test();cout << *p << endl;		// 可能会出错cout << *p << endl;		p = test();system("pause");
}

堆区

  • 由程序员分配和释放,如果不人为操作,则程序执行完之后由操作系统回收。
  • 主要利用new在堆区开辟内存。
    在这里插入图片描述
p本身也是局部变量,但是其存放的数据在堆区
code:#include<iostream>using namespace std;int* test(){int *p = new int(10);cout << "p指向的地址是:" << p << endl;return p;}void main(){int* p1 = test();cout << *p1 << endl;cout << *p1 << endl;cout << "p1指向的地址是:" << p1 << endl;delete p1;//cout << *p1 << endl;		//会报错,因为该地址已经被释放system("pause");}
result:p指向的地址是:000002A94D3560901010p1指向的地址是:000002A94D356090

内存开辟和释放

  • new开辟,delete释放
  • 类型*p = new 类型(初始值) ,前后类型要一致
  • delete[] p

在堆区开辟数组

  • new 类型[数组元素个数], 返回的是连续空间的首地址。
#include<iostream>
using namespace std;void test()
{int *array = new int[5];for (int i = 0; i < 5; i++){array[i] = i;}for (int i = 0; i < 5; i++){cout << array[i] << endl;}delete[] array;//cout << array[0] << endl;		//报错
}void main()
{test();system("pause");
}
http://www.lryc.cn/news/130738.html

相关文章:

  • [管理与领导-30]:IT基层管理者 - 人的管理 - 向上管理,管理好你的上司,职业发展事半功倍。什么样的上司不值得跟随?
  • Java进阶篇--迭代器模式
  • 【CAM】CAM(Class Activation Mapping)——可视化CNN的特征定位
  • Maven教程_编程入门自学教程_菜鸟教程-免费教程分享
  • Gof23设计模式之模板方法模式
  • springBoot 配置文件 spring.resources.add-mappings 参数的作用
  • 《Java极简设计模式》第03章:工厂方法模式(FactoryMethod)
  • C++11并发与多线程笔记(11) std::atomic续谈、std::async深入谈
  • React快速入门
  • windows权限维持—SSPHOOKDSRMSIDhistorySkeletonKey
  • CSS 两栏布局和三栏布局的实现
  • 消息中间件相关面试题
  • 成集云 | 电子签署集成腾讯云企业网盘 | 解决方案
  • 提升大数据技能,不再颓废!这6家学习网站是你的利器!
  • uniapp开发小程序-有分类和列表时,进入页面默认选中第一个分类
  • 小程序-uni-app:hbuildx uni-app 安装 uni-icons 及使用
  • PHP中in_array()函数用法详解
  • 热电联产在综合能源系统中的选址定容研究(matlab代码)
  • 校园网安全风险分析
  • kafka--kafka的基本概念-topic和partition
  • 【LVS】3、LVS+Keepalived群集
  • 对前端PWA应用的部分理解和基础Demo
  • CSGO饰品价格会一直下跌吗?市场何时止跌回升?
  • 线程池原理
  • 拷贝构造函数
  • 数据库: MySQL安装部署、主从
  • Java IO流(二)IO模型(BIO|NIO|AIO)
  • java版本spring cloud 企业工程系统管理 工程项目管理系统源码em
  • 飞天使-k8s简单搭建
  • java中把一个list转tree的方法