C++学习 --内存四区

2023-11-02 01:30
文章标签 c++ 学习 内存 四区

本文主要是介绍C++学习 --内存四区,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

1, 内存四区

1-1, 代码区

1-1-2, 特点

1-2, 全局区

1-2-1, 全局变量

1-2-2, 静态变量

1-2-3, 全局常量

1-3, 栈区

1-4, 堆区

1-4-1, new单个对象

1-4-2, delete单个对象

1-4-3, new&delete数组


1, 内存四区

四区中的数据,具有不同生命周期

1-1, 代码区

代码区存放的是编译后的C++代码二进制文件

1-1-2, 特点

特点共享、只读, 共享是只内存中的该段二进制文件可以反复执行,只读是指存放后的二进制文件不能被修改

1-2, 全局区

全局区存放的是C++代码的全局变量, 静态变量, 全局常量, 字符串常量的地方, 程序结束后由操作系统释放

1-2-1, 全局变量

全局变量为在函数外定义的变量

#include <iostream>
#include <string>using namespace std;/*全局变量存放在全局区,程序运行结束后由操作系统释放*///定义全局变量
int global_a = 10;
int global_b = 10;int main()
{int local_a = 10;int local_b = 20;cout << "局部变量的地址local_a:" << &local_a <<endl;cout << "局部变量的地址local_b:" << &local_b <<endl;cout << "全局变量的地址global_a:" << &global_a << endl;cout << "全局变量的地址global_b:" << &global_b << endl;system("pause");return 0;
}------------------------------------------------------------------
输出结果:
局部变量的地址local_a:00000045BEB5F9E4
局部变量的地址local_b:00000045BEB5FA04
//可以看出全局变量的地址与局部变量的地址不在一片内存空间
全局变量的地址global_a:00007FF74F00D000
全局变量的地址global_b:00007FF74F00D004

1-2-2, 静态变量

通过static 数据类型  变量名方式定义的变量为静态变量

#include <iostream>
#include <string>using namespace std;/*静态变量存放在全局区,程序运行结束后由操作系统释放*/int global_a = 10;
int global_b = 10;int main()
{//定义静态变量static int local_static_a = 10;static int local_static_b = 20;cout << "全局变量的地址global_a:" << & global_a << endl;cout << "全局变量的地址global_b:" << &global_b << endl;cout << "静态变量的地址local_static_a:" << &local_static_a << endl;cout << "静态变量的地址local_static_b:" << &local_static_b << endl;system("pause");return 0;
}
-----------------------------------------------------
输出结果:
//可以看出全局和静态变量存放在同一块内存区中
全局变量的地址global_a:00007FF60391D074
全局变量的地址global_b:00007FF60391D078
静态变量的地址local_static_a:00007FF60391D07C
静态变量的地址local_static_b:00007FF60391D080

1-2-3, 全局常量

在函数体外通过const 数据类型 变量名, 可定义一个全局常量

#include <iostream>
#include <string>using namespace std;/*全局常量,字符串常量存放在全局区,程序运行结束后由操作系统释放*/int global_a = 10;
int global_b = 10;//定义全局常量
const int c_global_a = 10;
const int c_global_b = 10;int main()
{//定义局部常量const int local_a = 10;const int local_b = 20;cout << "全局变量的地址global_a:" << & global_a << endl;cout << "全局变量的地址global_b:" << &global_b << endl;cout << "全局常量的地址c_global_c:" << &c_global_a << endl;cout << "全局常量的地址c_global_d:" << &c_global_b << endl;//&"aaaaa"表示获取该字符串常量的地址cout << "字符串常量的地址:" << &"aaaaa" << endl;cout << "局部常量的地址:" << &local_a << endl;cout << "局部常量的地址:" << &local_b << endl;system("pause");return 0;
}
----------------------------------------------------------------
输出结果:
//可以看出, 全局变量, 全局常量在同一块内存空间
全局变量的地址global_a:00007FF60F63D074
全局变量的地址global_b:00007FF60F63D078
全局常量的地址c_global_c:00007FF60F63ABB0
全局常量的地址c_global_d:00007FF60F63ABB4
字符串常量的地址:00007FF60F63ACA8
//可以看出局部常量与上面的常量不在一块内存空间
局部常量的地址:0000004A3B11F634
局部常量的地址:0000004A3B11F654

1-3, 栈区

栈区存放的是C++代码的函数的参数, 局部变量, 由编译器自动分配和释放。

注意:不要放回局部变量的地址, 因为局部变量在函数调用完后, 就被编译器释放。 

#include <iostream>
#include <string>using namespace std;/*局部变量和形参都是存放在栈区,由编译器分配和释放*/
int * func(int b)
{int a = 10;return &a;
}int main()
{int* p = func();cout << "调用func:" << *p << endl;system("pause");return 0;
}
---------------------------------------------------
输出结果:
//因为局部变量调用完后, 就被释放了, 所以这里没有获取到局部变量a的值
//同理形参b也一样
第一次调用func:-858993460

1-4, 堆区

堆区是由程序员分配和释放的, 若程序员不主动释放,在程序结束后由操作系统释放

1-4-1, new单个对象

通过new 数据类型(对象), 可在堆区创建一个地址,对象为地址指向的值, 返回指定数据类型的指针

#include <iostream>
#include <string>using namespace std;/*堆区的数据是由程序员分配的,若不主动释放,在程序
运行完成后, 由操作系统释放*/int * func()
{//通过new方式在堆区创建一块地址,返回对应数据类型的指针//地址存的数据是10//局部变量p,任然在栈区int* p = new int(10);return p;
}int main()
{int* p = func();cout << "调用func:" << *p << endl;cout << "调用func:" << *p << endl;system("pause");return 0;
}
-----------------------------------------------------
//能打印出10, 是因为10是存放在堆区上, 不会因为func调用完成后而释放
调用func:10
调用func:10

1-4-2, delete单个对象

通过delete 指针可手动释放堆区指定的内存空间

#include <iostream>
#include <string>using namespace std;/*堆区的数据是由程序员分配的,若不主动释放,在程序
运行完成后, 由操作系统释放*/int * func()
{//通过new方式在堆区创建一块地址,地址存的数据是10//局部变量p,任然在栈区int* p = new int(1011);return p;
}int main()
{int* p = func();cout << "调用func:" << *p << endl;//通过delete手动释放堆区的地址delete p;//该行调用,会报错,因为p指向的值已经被释放cout << "调用func:" << *p << endl;system("pause");return 0;
}
-----------------------------------------------------------
输出结果:
调用func:1011
//这里没有输出的因为是因为p已经被手动释放
调用func:

1-4-3, new&delete数组

通过new 数据类型[长度], 可在堆区创建一块数组地址空间, 同delete[] 指针, 手动释放创建的地址空间

#include <iostream>
#include <string>using namespace std;int * func()
{//表示在堆区中创建一个大小为10数组地址空间, 并赋值给指针arrint* arr = new int[10];//数组赋值for (int i = 0; i < 10; i++){arr[i] = i;}return arr;
}int main()
{int* arr = func();//逐个打印数组元素for (int i = 0; i < 10; i++){cout << arr[i] << " ";}cout << endl;//delete[]表示释放的是一个数组地址空间delete[] arr;system("pause");return 0;
}
-------------------------------------------------------------------
输出结果:
0 1 2 3 4 5 6 7 8 9

这篇关于C++学习 --内存四区的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/327155

相关文章

Java内存分配与JVM参数详解(推荐)

《Java内存分配与JVM参数详解(推荐)》本文详解JVM内存结构与参数调整,涵盖堆分代、元空间、GC选择及优化策略,帮助开发者提升性能、避免内存泄漏,本文给大家介绍Java内存分配与JVM参数详解,... 目录引言JVM内存结构JVM参数概述堆内存分配年轻代与老年代调整堆内存大小调整年轻代与老年代比例元空

从入门到精通C++11 <chrono> 库特性

《从入门到精通C++11<chrono>库特性》chrono库是C++11中一个非常强大和实用的库,它为时间处理提供了丰富的功能和类型安全的接口,通过本文的介绍,我们了解了chrono库的基本概念... 目录一、引言1.1 为什么需要<chrono>库1.2<chrono>库的基本概念二、时间段(Durat

C++20管道运算符的实现示例

《C++20管道运算符的实现示例》本文简要介绍C++20管道运算符的使用与实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录标准库的管道运算符使用自己实现类似的管道运算符我们不打算介绍太多,因为它实际属于c++20最为重要的

Visual Studio 2022 编译C++20代码的图文步骤

《VisualStudio2022编译C++20代码的图文步骤》在VisualStudio中启用C++20import功能,需设置语言标准为ISOC++20,开启扫描源查找模块依赖及实验性标... 默认创建Visual Studio桌面控制台项目代码包含C++20的import方法。右键项目的属性:

c++中的set容器介绍及操作大全

《c++中的set容器介绍及操作大全》:本文主要介绍c++中的set容器介绍及操作大全,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录​​一、核心特性​​️ ​​二、基本操作​​​​1. 初始化与赋值​​​​2. 增删查操作​​​​3. 遍历方

解析C++11 static_assert及与Boost库的关联从入门到精通

《解析C++11static_assert及与Boost库的关联从入门到精通》static_assert是C++中强大的编译时验证工具,它能够在编译阶段拦截不符合预期的类型或值,增强代码的健壮性,通... 目录一、背景知识:传统断言方法的局限性1.1 assert宏1.2 #error指令1.3 第三方解决

C++11委托构造函数和继承构造函数的实现

《C++11委托构造函数和继承构造函数的实现》C++引入了委托构造函数和继承构造函数这两个重要的特性,本文主要介绍了C++11委托构造函数和继承构造函数的实现,具有一定的参考价值,感兴趣的可以了解一下... 目录引言一、委托构造函数1.1 委托构造函数的定义与作用1.2 委托构造函数的语法1.3 委托构造函

C++11作用域枚举(Scoped Enums)的实现示例

《C++11作用域枚举(ScopedEnums)的实现示例》枚举类型是一种非常实用的工具,C++11标准引入了作用域枚举,也称为强类型枚举,本文主要介绍了C++11作用域枚举(ScopedEnums... 目录一、引言二、传统枚举类型的局限性2.1 命名空间污染2.2 整型提升问题2.3 类型转换问题三、C

C++链表的虚拟头节点实现细节及注意事项

《C++链表的虚拟头节点实现细节及注意事项》虚拟头节点是链表操作中极为实用的设计技巧,它通过在链表真实头部前添加一个特殊节点,有效简化边界条件处理,:本文主要介绍C++链表的虚拟头节点实现细节及注... 目录C++链表虚拟头节点(Dummy Head)一、虚拟头节点的本质与核心作用1. 定义2. 核心价值二

C++ 检测文件大小和文件传输的方法示例详解

《C++检测文件大小和文件传输的方法示例详解》文章介绍了在C/C++中获取文件大小的三种方法,推荐使用stat()函数,并详细说明了如何设计一次性发送压缩包的结构体及传输流程,包含CRC校验和自动解... 目录检测文件的大小✅ 方法一:使用 stat() 函数(推荐)✅ 用法示例:✅ 方法二:使用 fsee