〇、前言

本文主要围绕 .Net 框架中的托管堆(Heap,简称堆)和堆栈(Stack,简称栈)展开。

.Net 程序在 CLR(Common Language Runtime 公共语言运行时)上运行时,内存被从逻辑上划分为两个主要部分:堆和栈。除了栈和堆之外,CLR 还维护了其他一些内存区域,例如静态存储区域(Static Storage Area)、常量存储区域(Constant Storage Area)等。这些内存区域都有各自的特点和用途,可以帮助我们更好地管理程序内存和资源的使用。

因此,熟知堆和栈的运行机制,对提升系统性能和稳定性至关重要。

一、值类型和引用类型

在介绍主角之前我们先来了解一下值类型和引用类型。

1.1 值类型

先看下都有哪些类型属于值类型:

类别 例举
整型数值类型
sbyte(-128~127)、byte(0~255)、
short(-32768~32767)、ushort(0~65535)、
int(-2147483648~2147483647)、uint(0~2147483647)、
long(-9,233,372,036,854,775,808~9,223,372,036,854,775,807)、ulong(0~18,446,744,073,709,551,615)、
nint(取决于在运行时计算的平台:带符号的 32 位或 64 位整数)、
nuint(取决于在运行时计算的平台:无符号的 32 位或 64 位整数)
浮点型数值类型
float(f/F)(2.77f、2.77F、1_000.000_012f、1_000.000_012F)
double(d/D)(3d、3D、3.000_012) // 可以不用带字母
decimal(m/M)(3_000.5m、400.75M)
布尔类型
bool check = true; // true / false
Unicode UTF-16 字符
char(U+0000~U+FFFF)
(三种方法指定 char 值:
1. 字符文本;
2. Unicode 转义序列,它是\u后跟四位字符代码的十六进制表示形式;
3. 十六进制转义序列,它是 \x 后跟字符代码的十六进制表示形式)
枚举类型
// 默认情况下,枚举成员的关联常数值为类型 int;它们从零开始,并按定义文本顺序递增 1。
// 也可以显式指定任何其他整数数值类型作为枚举类型的基础类型。
// 还可以显式指定关联的常数值,如下示例:
enum ErrorCode : ushort
{
None = 0,
Unknown = 1,
ConnectionLost = 100,
OutlierReading = 200
}
结构类型(structure/struct type)
// 结构类型具有值语义。也就是说,结构类型的变量包含类型的实例。
// 默认情况下,在分配中,通过将参数传递给方法并返回方法结果来复制变量值。
// 对于结构类型变量,将复制该类型的实例。
public struct Coords
{
public Coords(double x, double y)
{
X = x;
Y = y;
}
public double X { get; }
public double Y { get; }
public override string ToString() => $"({X}, {Y})";
}
// 可以使用结构类型来设计以数据为中心的较小类型,这些类型只有很少的行为或没有行为。
// 例如,.NET 使用结构类型来表示数字(整数和实数)、布尔值、Unicode 字符以及时间实例。
元组类型
// 元组功能提供了简洁的语法来将多个数据元素分组成一个轻型数据结构。
(double Sum, int Count) t2 = (4.5, 3);
Console.WriteLine($"Sum of {t2.Count} elements is {t2.Sum}.");
// Output:
// Sum of 3 elements is 4.5. // 如前面的示例所示,若要定义元组类型,需要指定其所有数据成员的类型,或者,可以指定字段名称。
// 虽然不能在元组类型中定义方法,但可以使用 .NET 提供的方法,如下面的示例所示
(double, int) t = (4.5, 3);
Console.WriteLine(t.ToString());
Console.WriteLine($"Hash code of {t} is {t.GetHashCode()}.");
// Output:
// (4.5, 3)
// Hash code of (4.5, 3) is 718460086.

1.2 引用类型

再来看下哪些属于引用类型:

类别 例举
内置引用类型
object 对象类型。可以将任何类型的值赋给 object 类型的变量
string 字符串类型。string 类型表示零个或多个 Unicode 字符的序列。定义 == 和 != 是为了比较 string 对象(而不是引用)的值
string a = "hello";
string b = "h";
// Append to contents of 'b'
b += "ello";
Console.WriteLine(a == b); // true
Console.WriteLine(object.ReferenceEquals(a, b)); // false
string 字符串文本。三种类型:原始、带引号、逐字。
原始字符串,就是至少三个双引号 (""") 括起来,"且自动忽略最左侧的空格:""" This is a text.""";(C# 11 可用)
带引号字符串,就是常用的双引号方式:"This is a text.";
逐字字符串,就是在双引号前加符号 @:@"This is a text."
UTF-8 字符串。 从 C# 11 开始,可以将 u8 后缀添加到字符串字面量以指定 UTF-8 编码。 UTF-8 字面量存储为 ReadOnlySpan<byte> 对象
ReadOnlySpan<byte> AuthWithTrailingSpace = new byte[] { 0x41, 0x55, 0x54, 0x48, 0x20 };
ReadOnlySpan<byte> AuthStringLiteral = "AUTH "u8;
byte[] AuthStringLiteral = "AUTH "u8.ToArray();
委托类型。它有一个返回值和任意数目任意类型的参数,如下示例:
public delegate void MessageDelegate(string message);
public delegate int AnotherDelegate(MyType m, long num);
Action<string> stringAction = str => {};
Action<object> objectAction = obj => {};
// Creates a new delegate instance with a runtime type of Action<string>.
Action<string> wrappedObjectAction = new Action<string>(objectAction);
// The two Action<string> delegate instances can now be combined.
Action<string> combination = stringAction + wrappedObjectAction;
动态类型。dynamic 类型表示变量的使用和对其成员的引用绕过编译时类型检查,改为在运行时解析这些操作。
static void Main(string[] args)
{
dynamic dyn = 1;
object obj = 1;
System.Console.WriteLine(dyn.GetType());
// dyn = dyn + 3;
// obj = obj + 3; // 编译错误
System.Console.WriteLine(obj.GetType());
}
// 对于 dyn + 3,不会报告任何错误。 在编译时不会检查包含 dyn 的表达式,原因是 dyn 的类型为 dynamic
记录
// 从 C# 9 开始,可以使用 record 修饰符定义一个引用类型,用来提供用于封装数据的内置功能
// C# 10 允许 record class 语法作为同义词来阐明引用类型,并允许 record struct 使用相同功能定义值类型
public record Person(string FirstName, string LastName);
public static void Main()
{
Person person = new("Nancy", "Davolio");
Console.WriteLine(person);
// output: Person { FirstName = Nancy, LastName = Davolio }
}
// 使用 class 关键字声明类
class TestClass
{
// Methods, properties, fields, events, delegates
// and nested classes go here.
}
接口
// 接口定义协定。实现该协定的任何 class 或 struct 必须提供接口中定义的成员的实现
interface ISampleInterface // 定义接口
{
void SampleMethod();
}
class ImplementationClass : ISampleInterface // 实现接口
{
void ISampleInterface.SampleMethod()
{
// 执行的操作详情
}
static void Main()
{
ISampleInterface obj = new ImplementationClass(); // 声明一个接口的实例
obj.SampleMethod(); // 调用接口方法
}
}

1.3 值类型与引用类型的对比

  • 值类型变量声明后,不管是否已经赋值,编译器为其分配内存。
  • 引用类型当声明一个类时,只在栈中分配一小片内存用于容纳一个地址,而此时并没有为其分配堆上的内存空间。当使用 new 创建一个类的实例时,分配堆上的空间,并把堆上空间的地址保存到栈上分配的小片空间中。
  • 值类型的实例通常是在线程栈上分配的(静态分配),但是在某些情形下可以存储在堆中。
  • 引用类型的对象总是在进程堆中分配(动态分配)。
  • 值类型的变量直接包含类型的实际值。
  • 引用类型包含对类型实例的引用。
  • 两个不同的变量不能指向同一个值类型的值。
  • 两个不同的变量可以指向同一个引用类型的值,只是这两个变量存的内存地址引用相同。

参考:https://learn.microsoft.com/zh-cn/dotnet/csharp/language-reference/builtin-types/value-types

二、关于堆和栈的理解

2.1 堆和栈的概念

堆(托管堆和非托管堆)

  托管堆:

  其实在 C 语言中才叫堆,C# 中叫托管堆。托管堆是由 CLR(公共语言运行库 Common Language Runtime)管理,当堆中满了之后,会自动清理堆中的垃圾,因此加了“托管”两字。这也是为什么 .Net 开发不需要关心内存释放的原因。

  托管堆是一块动态分配的内存区域,用于存储程序运行时需要的数据。当声明一个引用类型对象或变量时,它们被分配到堆上,并返回其引用(即指向该对象或变量在堆中存储位置的指针)。堆中的对象或变量可以通过其引用来访问和修改。  

  非托管堆:

  .NET的程序还包含了非托管的堆,所有需要分配堆内存的非托管资源将会被分配到非托管堆上。非托管的堆需要程序员用指针手动地分配,并且手动地释放,.NET的垃圾回收和内存管理制度不适用于非托管的堆。

  常见的非托管资源有:文件流、图像图形类、数据库的连接,网络连接,系统的窗口句柄,打印机资源等。

  

  栈是一种基于后进先出(Last In First Out,LIFO)原则的内存区域。栈存储几种类型的数据:某些类型变量的值、程序当前的执行环境、传递给方法的参数。当程序调用一个方法时,该方法的参数、返回地址和局部变量等数据会被压入栈中。当方法执行结束时,这些数据会从栈中弹出。

  本质上讲堆栈也是一种线性结构,符合线性结构的基本特点:即每个节点有且只有一个前驱节点和一个后续节点。

  栈把所有操作限制在“只能在线性结构的某一端”进行,而不能在中间插入或删除元素。把数据放入栈顶称为入栈(push), 从栈顶删除数据称为出栈(pop)。

  

2.2 堆和栈的关系

内存空间分配

  • 堆:开发人员可以根据实际需要进行动态扩展堆的大小,但过多的使用堆,会影响程序性能和稳定性。
  • 栈:栈的大小一般比较有限,不能在栈上存储过多的数据,一般用于存放函数的参数值、局部值类型变量的值、引用类型的引用等。

缓存方式

  • 堆:存放在二级缓存中,生命周期由垃圾回收算法来决定,并非弃用后立马被回收,所以调用这些对象的速度要相对来得低一些。
  • 栈:使用的是一级缓存, 他们通常都是被调用时从栈顶取出,用完即释放。

数据结构

  • 堆:堆中一般保存的是实际的数据值,例如引用类型的实例值。
  • 栈:栈中一般存储较小体量的值,例如值类型的值、引用类型的引用等。

存取排列方式

  • 堆:堆里的内存能够以任意顺序存入和移除。
  • 栈:LIFO 后进先出。必须从栈顶进行存取操作。

三、堆栈是如何配合来保证程序的正常运行的?

在.NET中,堆栈(stack)、托管堆(managed heap)、非托管堆(unmanaged heap)和垃圾回收机制配合使用来保证程序的正常运行。以下是它们之间的协同工作方式:

  • 堆栈:堆栈用于管理函数调用和局部变量,确保函数的正确执行和返回。每个线程都有一个私有的堆栈,用于存储函数调用的上下文信息。当一个函数被调用时,相关的信息会被压入堆栈;而当函数执行结束后,这些信息会从堆栈中弹出。堆栈的快速分配和释放特性使得函数调用能够高效地进行。
  • 托管堆:托管堆是.NET中用于存储托管对象的内存区域,由 CLR(Common Language Runtime)负责管理。所有的托管对象都分配在托管堆上,并由 CLR 自动进行内存管理。当需要创建一个对象时,CLR 会在托管堆上为其分配内存;而当对象不再被引用时,垃圾回收机制会定期清理无用的对象,并回收它们所占用的内存空间,防止内存泄漏和资源浪费。
  • 非托管堆:非托管堆是指 .NET 应用程序使用的非托管资源的内存区域,例如通过使用平台调用接口(P/Invoke)调用的外部库或操作系统资源。非托管堆通常由操作系统或外部库分配和管理,而 .NET 应用程序通过与非托管代码进行交互来访问和操作非托管堆。
  • 垃圾回收机制:垃圾回收器是 CLR 中的组件,负责管理托管堆中的内存。它会周期性地扫描托管堆,标记并清理不再被引用的对象,并释放其占用的内存空间。垃圾回收机制通过追踪对象之间的引用关系,确定哪些对象可以被安全地回收,以避免内存泄漏和资源浪费。

堆栈管理函数调用和局部变量,托管堆管理托管对象的分配和回收,非托管堆用于存储和管理非托管资源,而垃圾回收机制定期清理无用的对象,释放内存资源。这种机制确保了程序的内存使用效率和稳定性,减轻了开发人员对内存管理的负担,使他们能够更专注于应用程序的逻辑开发。

参考:https://blog.csdn.net/qq_44034384/article/details/106611384  https://qianwen.aliyun.com/chat  https://blog.csdn.net/beenles/article/details/130710732

堆 Heap & 栈 Stack(.Net)【概念解析系列_3】【C# 基础】的更多相关文章

  1. (转)Java里的堆(heap)栈(stack)和方法区(method)(精华帖,多读读)

    [color=red][/color]<一> 基础数据类型直接在栈空间分配, 方法的形式参数,直接在栈空间分配,当方法调用完成后从栈空间回收.   引用数据类型,需要用new来创建,既在栈 ...

  2. Java里的堆(heap)栈(stack)和方法区(method)

    基础数据类型直接在栈空间分配, 方法的形式参数,直接在栈空间分配,当方法调用完成后从栈空间回收.   引用数据类型,需要用new来创建,既在栈空间分配一个地址空间,又在堆空间分配对象的类变量 . 方法 ...

  3. .NET的堆和栈01,基本概念、值类型内存分配

    当我们对.NET Framework的一些基本面了解之后,实际上,还是很有必要了解一些更底层的知识.比如.NET Framework是如何进行内存管理的,是如何垃圾回收的......这样,我们才能写出 ...

  4. JVM调优总结(一)-- 堆和栈的基本概念

    数据类型 Java虚拟机中,数据类型可以分为两类:基本类型和引用类型.基本类型的变量保存原始值,即:他代表的值就是数值本身:而引用类型的变量保存引用值.“引用值”代表了某个对象的引用,而不是对象本身, ...

  5. 数据结构11: 栈(Stack)的概念和应用及C语言实现

    栈,线性表的一种特殊的存储结构.与学习过的线性表的不同之处在于栈只能从表的固定一端对数据进行插入和删除操作,另一端是封死的. 图1 栈结构示意图 由于栈只有一边开口存取数据,称开口的那一端为“栈顶”, ...

  6. (转)堆heap和栈stack

    一 英文名称 堆和栈是C/C++编程中经常遇到的两个基本概念.先看一下它们的英文表示: 堆――heap 栈――stack 二 从数据结构和系统两个层次理解 在具体的C/C++编程框架中,这两个概念并不 ...

  7. NET的堆和栈04,对托管和非托管资源的垃圾回收以及内存分配

    在" .NET的堆和栈01,基本概念.值类型内存分配"中,了解了"堆"和"栈"的基本概念,以及值类型的内存分配.我们知道:当执行一个方法的时 ...

  8. .NET的堆和栈03,引用类型对象拷贝以及内存分配

    在" .NET的堆和栈01,基本概念.值类型内存分配"中,了解了"堆"和"栈"的基本概念,以及值类型的内存分配.我们知道:当执行一个方法的时 ...

  9. .NET的堆和栈02,值类型和引用类型参数传递以及内存分配

    在" .NET的堆和栈01,基本概念.值类型内存分配"中,了解了"堆"和"栈"的基本概念,以及值类型的内存分配.我们知道:当执行一个方法的时 ...

  10. C#中的堆和栈

    看到一篇讲堆和栈的文章,是我目前为止见到讲的最易懂,详细和深入的.我翻译成中文.以此总结. 原文=>C#Heap(ing) Vs Stack(ing) in .NET 在net framewor ...

随机推荐

  1. Azure DevOps(三)Azure Pipeline 自动化将程序包上传到 Azure Bolb Storage

    一,引言 结合前几篇文章,我们了解到 Azure Pipeline 完美的解决了持续集成,自动编译.同时也兼顾了 Sonarqube 作为代码扫描工具.接下来另外一个问题出现了,Azure DevOp ...

  2. 2020-12-02:mysql中,一张表里面有 ID 自增主键,当 insert 了 17 条记录之后,删除了第 15,16,17 条记录,再把 Mysql 重启,再 insert 一条记录,这条记

    2020-12-02:mysql中,一张表里面有 ID 自增主键,当 insert 了 17 条记录之后,删除了第 15,16,17 条记录,再把 Mysql 重启,再 insert 一条记录,这条记 ...

  3. 【GiraKoo】C++中static关键字的作用

    C++中static关键字的作用 在程序中良好的使用static,const,private等关键字,对于代码的健壮性有很大的帮助. 本文介绍的就是C++中static关键字的一些常见用法与区别.适合 ...

  4. 2019年蓝桥杯C/C++大学B组省赛真题(数的分解)

    题目描述: 把2019分解成3个各不相同的正整数之和,并且要求每个正整数都不包含数字2和4,一共有多少种不同的分解方法? 注意交换3个整数的顺序被视为同一种方法,例如1000+1001+18 和100 ...

  5. wait,notify,notifyAll,sleep,join等线程方法的全方位演练

    一.概念解释 1. 进入阻塞: 有时我们想让一个线程或多个线程暂时去休息一下,可以使用 wait(),使线程进入到阻塞状态,等到后面用到它时,再使用notify().notifyAll() 唤醒它,线 ...

  6. Docker化Spring Boot应用

    本文翻译自国外论坛 medium,原文地址:https://medium.com/@bubu.tripathy/dockerizing-your-spring-boot-application-75b ...

  7. Spring配置动态数据库

    前言 本文主要介绍使用spring boot 配置多个数据库,即动态数据库 开始搭建 首先创建一个SpringWeb项目--dynamicdb(spring-boot2.5.7) 然后引入相关依赖lo ...

  8. [ESP] 私有版Rainmaker User Mapping

    [ESP] 私有版Rainmaker User Mapping 1. 设备烧录的程序esp-rainmaker/examples/gpio这个demo 我这里是自己的工程,可以参照 idf.py se ...

  9. ARC114F Permutation Division

    题意 给定一个 \(1 \sim N\) 的排列,Alice 把它划分成 \(k\) 段,Bob 把这 \(k\) 段任意排列.Alice 想让字典序最小,Bob 想让字典序最大.请问最后的排列. 数 ...

  10. tvm-多线程代码生成和运行

    本文链接 https://www.cnblogs.com/wanger-sjtu/p/16818492.html 调用链 tvm搜索算子在需要多线程运行的算子,是在codegen阶段时插入TVMBac ...