• 【微软技术栈】使用新的C#功能减少内存分配


    本文内容

    1. 通过引用传递和返回
    2. 引用安全上下文
    3. 安全的上下文和 ref 结构
    4. 统一内存类型
    5. 通过参考安全提高性能

    本节中介绍的技术可提高应用于代码中的热路径时的性能。热路径是代码库中在正常操作中经常重复执行的部分。将这些技术应用于不经常执行的代码将产生最小的影响。在进行任何更改以提高性能之前,测量基线至关重要。然后,分析该基线以确定内存瓶颈发生的位置。

    测量内存使用情况并确定可以减少分配后,请使用本节中的技术来减少分配。每次连续更改后,再次测量内存使用情况。确保每个更改都会对应用程序中的内存使用产生积极影响。

    .NET 中的性能工作通常意味着从代码中删除分配。您分配的每个内存块最终都必须被释放。较少的分配可减少垃圾回收所花费的时间。它通过从特定代码路径中删除垃圾回收,允许更可预测的执行时间。

    减少分配的常用策略是将关键数据结构从类型更改为类型。此更改会影响使用这些类型的语义。参数和返回现在按值传递,而不是按引用传递。如果类型很小,只有三个字或更少(考虑到一个字的自然大小为一个整数),则复制值的成本可以忽略不计。它是可衡量的,可以对较大的类型产生真正的性能影响。为了消除复制的影响,开发人员可以传递这些类型来获取预期的语义。

    使用 C# 功能,您可以表达所需的类型语义,而不会对其整体可用性产生负面影响。在实现这些增强功能之前,开发人员需要求助于具有指针和原始内存的构造,以实现相同的性能影响。编译器为新的相关功能生成可验证的安全代码可验证的安全代码意味着编译器检测到可能的缓冲区溢出或访问未分配或释放的内存。编译器会检测并防止某些错误。

    1、通过引用传递和返回

    C# 中的变量存储。在类型中,该值是该类型的实例的内容。在类型中,该值是对存储该类型实例的内存块的引用。添加修饰符意味着变量存储对值的引用。在类型中,引用指向包含该值的存储。在类型中,引用指向包含对内存块的引用的存储。

    在 C# 中,方法的参数是按值传递的,返回值是按值返回的。参数的值将传递给该方法。返回参数的值是返回值。

    或 修饰符指示参数是通过引用传递的。对存储位置的引用将传递给该方法。添加到方法签名意味着通过引用返回返回值。对存储位置的引用是返回值。

    您还可以使用 ref 赋值让一个变量引用另一个变量。典型的赋值将右侧的值复制到赋值左侧的变量。ref 赋值将右侧变量的内存位置复制到左侧的变量。现在指的是原始变量:

    1. int anInteger = 42; // assignment.
    2. ref int location = ref anInteger; // ref assignment.
    3. ref int sameLocation = ref location; // ref assignment
    4. Console.WriteLine(location); // output: 42
    5. sameLocation = 19; // assignment
    6. Console.WriteLine(anInteger); // output: 19

    分配变量时,会更改其。当您 ref 赋值一个变量时,您可以更改它所引用的内容。

    您可以使用变量、通过引用传递和引用赋值直接处理值的存储。编译器强制执行的范围规则可确保直接使用存储时的安全性。

    和修饰符都指示参数应通过引用传递,并且不能在方法中重新赋值。区别在于,该方法将参数用作变量。该方法可能会捕获参数,也可能通过只读引用返回参数。在这些情况下,应使用修饰符。否则,修饰符将提供更大的灵活性。您无需将修饰符添加到参数的参数中,因此您可以使用修饰符安全地更新现有 API 签名。如果未将 or 修饰符添加到参数的参数中,编译器将发出警告。

    2、引用安全上下文

    C# 包含表达式规则,以确保在表达式引用的存储不再有效的情况下无法访问表达式。请看以下示例:

    1. public ref int CantEscape()
    2. {
    3. int index = 42;
    4. return ref index; // Error: index's ref safe context is the body of CantEscape
    5. }

    编译器报告错误,因为无法从方法返回对局部变量的引用。调用方无法访问所引用的存储。ref 安全上下文定义表达式可以安全访问或修改的范围。下表列出了变量类型的 ref 安全上下文。 字段不能在 a 或 non-ref 中声明,因此这些行不在表中:

    声明ref 安全上下文
    非 ref 本地声明 local 的块
    non-ref 参数current 方法
    ref参数ref readonlyin调用方法
    out参数current 方法
    class调用方法
    non-ref 字段structcurrent 方法
    ref领域ref struct调用方法

    如果变量的 ref 安全上下文是调用方法,则可以返回该变量。如果其 ref 安全上下文是当前方法或块,则不允许返回。以下代码片段显示了两个示例。可以从调用方法的作用域访问成员字段,因此类或结构字段的 ref 安全上下文是调用方法。带有 或修饰符的参数的 ref 安全上下文是整个方法。两者都可以从成员方法返回:

    1. private int anIndex;
    2. public ref int RetrieveIndexRef()
    3. {
    4. return ref anIndex;
    5. }
    6. public ref int RefMin(ref int left, ref int right)
    7. {
    8. if (left < right)
    9. return ref left;
    10. else
    11. return ref right;
    12. }

    编译器确保引用无法转义其引用安全上下文。您可以安全地使用参数、 和局部变量,因为编译器会检测您是否意外编写了代码,在表达式存储无效时可以访问表达式。

    3、安全的上下文和 ref 结构

    ref struct类型需要更多的规则来确保它们可以安全使用。类型可以包含字段。这需要引入一个安全的环境。对于大多数类型,安全上下文是调用方法。换言之,始终可以从方法返回不是 ref structrefref struct的值。

    非正式地,a 的安全上下文是可以访问其所有字段的范围。换句话说,它是其所有字段的 ref 安全上下文的交集。以下方法返回 a to a member 字段,因此其安全上下文是该方法:ref structrefrefReadOnlySpan

    1. private string longMessage = "This is a long message";
    2. public ReadOnlySpan<char> Safe()
    3. {
    4. var span = longMessage.AsSpan();
    5. return span;
    6. }

    相反,以下代码会发出错误,因为 的成员引用了堆栈分配的整数数组。它无法转义方法:ref fieldSpan

    1. public Span<int> M()
    2. {
    3. int length = 3;
    4. Span<int> numbers = stackalloc int[length];
    5. for (var i = 0; i < length; i++)
    6. {
    7. numbers[i] = i;
    8. }
    9. return numbers; // Error! numbers can't escape this method.
    10. }

    4、统一内存类型

    System.Span 和 System.Memory 的引入为使用内存提供了统一的模型。System.ReadOnlySpan<T> 和 System.ReadOnlyMemory 提供用于访问内存的只读版本。它们都提供了对存储类似元素数组的内存块的抽象。区别在于 和 是类型,而 和 是类型。跨度包含一个 .因此,跨度的实例不能离开其安全上下文。a 的安全上下文是其 的 ref 安全上下文。实施并删除此限制。您可以使用这些类型直接访问内存缓冲区。SpanReadOnlySpanref structMemoryReadOnlyMemorystructref fieldref structref fieldMemoryReadOnlyMemory

    5、通过参考安全提高性能

    使用这些功能提高性能涉及以下任务:

    • 避免分配:将类型从 a 更改为 时,会更改其存储方式。局部变量存储在堆栈上。分配容器对象时,成员以内联方式存储。此更改意味着分配更少,从而减少了垃圾回收器所做的工作。它还可能会降低内存压力,从而降低垃圾回收器的运行频率。
    • 保留引用语义:将类型从 a 更改为 a 会更改将变量传递给方法的语义。修改其参数状态的代码需要修改。现在参数是 ,该方法正在修改原始对象的副本。您可以通过将该参数作为参数传递来还原原始语义。更改后,该方法将再次修改原始内容。
    • 避免复制数据:复制较大的类型可能会影响某些代码路径的性能。还可以添加修饰符,以便通过引用而不是按值将较大的数据结构传递给方法。
    • 限制修改:当通过引用传递类型时,被调用的方法可以修改结构的状态。您可以将修饰符替换为 or 修饰符,以指示无法修改参数。当方法捕获参数或通过只读引用返回参数时,首选。还可以创建类型或具有成员的类型,以便更好地控制可以修改的成员。
    • 直接操作内存:当将数据结构视为包含一系列元素的内存块时,某些算法最有效。和类型提供对内存块的安全访问。

    这些技术都不需要代码。如果使用得当,您可以从安全代码中获得性能特征,而以前只能使用不安全技术才能实现。

  • 相关阅读:
    《Java极简设计模式》第08章:外观模式(Facade)
    Leetcode DAY 15: 层序遍 and 翻转二叉树 and 对称二叉树
    C语言练习百题之位符号&的使用
    【Web开发】纯前端实现科技企业官网首页
    Liunx安装Tomcat部署Java项目
    GPT提示语格式——个人自用
    【嵌入式开源学习】__发现了一个很棒的开源项目CSON
    游戏录屏软件推荐,教你录制高清游戏视频
    Windows11国内首发,宁盾终端准入率先实现兼容
    分布式IO系统BL201 Profinet耦合器
  • 原文地址:https://blog.csdn.net/m0_51887793/article/details/134541503