0

0

c# 如何避免在 lock 语句块内分配内存(GC.KeepAlive)

幻夢星雲

幻夢星雲

发布时间:2026-01-22 12:53:03

|

385人浏览过

|

来源于php中文网

原创

lock块内分配内存会引发GC压力和对象提早回收风险,因频繁创建临时对象及JIT误判生命周期;应将分配移出lock,复用缓冲区,禁用LINQ,慎用GC.KeepAlive仅限非托管资源场景。

c# 如何避免在 lock 语句块内分配内存(gc.keepalive)

为什么 lock 语句块里分配内存会引发问题

不是 lock 本身导致 GC 压力,而是锁内频繁创建临时对象(如字符串拼接、LINQ 查询、装箱、匿名函数闭包)会让 GC 频繁触发,尤其在高并发短临界区场景下,可能拖慢吞吐量或造成锁持有时间意外延长。更隐蔽的问题是:若锁保护的对象在 lock 块中被局部变量引用但未被后续代码使用,JIT 可能提前判定其“已死”,触发提早回收——哪怕锁还没释放,这在涉及非托管资源或回调注册时会出错。

哪些操作在 lock 中实际会分配内存

以下常见写法在 lock 块内执行时,几乎必然触发堆分配:

  • string.Concat(a, b, c)$"x={x} y={y}"(插值字符串在 .NET 6+ 有优化,但仍有逃逸可能)
  • list.Where(x => x > 0).ToList()(LINQ 构建迭代器 + 新 List)
  • new byte[1024]new Dictionary<int, string>()
  • 任何值类型到 object 的装箱(如 obj = i; 其中 iint
  • 捕获局部变量的 lambda:ThreadPool.QueueUserWorkItem(_ => DoWork(x))x 被闭包捕获并堆分配)

如何避免:从分配源头控制 + GC.KeepAlive 的真实用途

GC.KeepAlive 不是用来“防止 lock 内分配”的,它只解决一个特定问题:确保某个对象在代码执行到某一点之前**不会被 GC 回收**,常用于 P/Invoke 场景中托管对象与非托管句柄生命周期不同步的情况。它对减少内存分配完全无效。

真正可行的策略是:

  • 把内存分配移出 lock —— 先计算、构造、复制,再进锁做原子更新
  • 复用缓冲区:用 ArrayPool<byte>.Shared.Rent() 替代 new byte[n]
  • Span<T> / ReadOnlySpan<T> 处理字符串切片,避免子串分配
  • 禁用 LINQ,改用 for 循环 + 预分配集合(如 var results = new List<T>(capacity)
  • 若必须在锁内调用外部方法,确认该方法是否分配——必要时加注释或用 [MethodImpl(MethodImplOptions.AggressiveInlining)] 引导内联(仅适用于小函数)
private readonly object _sync = new();
private List<int> _data = new();

public void AddRange(ReadOnlySpan<int> items)
{
    // ✅ 分配在外:先拷贝到栈或复用池
    var temp = ArrayPool<int>.Shared.Rent(items.Length);
    try
    {
        items.CopyTo(temp);
        // ✅ 锁内只做最小原子操作
        lock (_sync)
        {
            _data.AddRange(temp.AsSpan(0, items.Length));
        }
    }
    finally
    {
        ArrayPool<int>.Shared.Return(temp);
    }
}

GC.KeepAlive 应该在什么时机用?别乱加

只有当你明确存在「对象在锁释放后仍需被非托管代码访问」的风险时才需要 GC.KeepAlive。典型场景是:你在 lock 块中调用了 Marshal.AllocHGlobal 并把指针传给了 native 代码,而托管对象(比如一个 byte[])只是用来管理这块内存的生命周期。

Bolt.new
Bolt.new

Bolt.new是一个免费的AI全栈开发工具

下载

此时错误写法是:

lock (_sync)
{
    var buffer = new byte[1024];
    IntPtr ptr = Marshal.AllocHGlobal(1024);
    CopyToNative(ptr, buffer); // 假设这个函数把 buffer 数据复制过去
    RegisterWithNative(ptr);   // native 侧开始异步使用 ptr
    // ❌ buffer 在这里就可能被 GC 回收,即使 native 还在用 ptr
}

正确做法是让 buffer 的生命周期至少延续到 native 使用结束之后:

lock (_sync)
{
    var buffer = new byte[1024];
    IntPtr ptr = Marshal.AllocHGlobal(1024);
    CopyToNative(ptr, buffer);
    RegisterWithNative(ptr);
    // ✅ 确保 buffer 不被提前回收,直到 native 完成
    GC.KeepAlive(buffer);
}

注意:GC.KeepAlive 必须放在所有依赖该对象的逻辑之后,且不能被 JIT 优化掉(它本质是插入一个“使用”屏障)。它和 lock 没有语法绑定关系,加在锁外、锁内、甚至锁后都行——关键是语义上要覆盖整个非托管使用期。

绝大多数业务代码根本不需要 GC.KeepAlive;滥用它只会掩盖设计缺陷,比如本该用 SafeHandle 封装的资源却手动管理生命周期。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

1030

2023.08.02

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

760

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

221

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1567

2023.10.24

字符串介绍
字符串介绍

字符串是一种数据类型,它可以是任何文本,包括字母、数字、符号等。字符串可以由不同的字符组成,例如空格、标点符号、数字等。在编程中,字符串通常用引号括起来,如单引号、双引号或反引号。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

649

2023.11.24

java读取文件转成字符串的方法
java读取文件转成字符串的方法

Java8引入了新的文件I/O API,使用java.nio.file.Files类读取文件内容更加方便。对于较旧版本的Java,可以使用java.io.FileReader和java.io.BufferedReader来读取文件。在这些方法中,你需要将文件路径替换为你的实际文件路径,并且可能需要处理可能的IOException异常。想了解更多java的相关内容,可以阅读本专题下面的文章。

1228

2024.03.22

php中定义字符串的方式
php中定义字符串的方式

php中定义字符串的方式:单引号;双引号;heredoc语法等等。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

1204

2024.04.29

go语言字符串相关教程
go语言字符串相关教程

本专题整合了go语言字符串相关教程,阅读专题下面的文章了解更多详细内容。

192

2025.07.29

C# ASP.NET Core微服务架构与API网关实践
C# ASP.NET Core微服务架构与API网关实践

本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

76

2026.03.11

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
国外Web开发全栈课程全集
国外Web开发全栈课程全集

共12课时 | 1万人学习

进程与SOCKET
进程与SOCKET

共6课时 | 0.4万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号