news 2026/5/1 5:00:49

告别频繁 GC:C#.NET PooledList 的设计与使用场景

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
告别频繁 GC:C#.NET PooledList 的设计与使用场景

简介

PooledList<T>是 高性能集合类型,由Collections.Pooled提供,用于替代List<T>,通过 对象池 (ArrayPool<T>) 复用内部数组来减少GC(垃圾回收)压力。

⚡ 核心目标:
在需要频繁创建/销毁List<T>的场景下,PooledList<T>通过数组租借与归还的机制避免频繁分配内存,从而提升性能并降低GC负担。

安装

dotnetaddpackage Collections.Pooled --version1.0.82

添加命名空间

usingCollections.Pooled;

特点

  • 数组池化:内部数组从ArrayPool<T>.Shared(默认)或自定义池中租借,减少分配。

  • Span<T>支持:提供Span属性,直接访问内部数组的填充部分,支持零拷贝操作。

  • IDisposable实现:调用Dispose()时,返回内部数组到池中(不调用Dispose不会出错,但会降低池化效果)。

  • 扩展方法:如TryFindTryFindLast(替换标准FindFindLast,返回bool以避免异常)。

  • 添加/插入SpanAddSpanInsertSpan方法返回一个Span<T>,允许直接写入内部存储,而无需多次调用Add

  • 构造函数选项:

    • 支持指定自定义ArrayPool<T>

    • clearMode参数控制数组返回池时是否清除内容(默认自动)。

    • sizeToCapacity参数使初始Count == Capacity,适合值类型避免不必要的零初始化。

  • ToPooledList()扩展:从IEnumerable<T>快速创建PooledList<T>

  • 性能提升:在高频操作中,减少GC触发,尤其适合循环中创建临时列表的场景。

内部原理

普通List<T>内部
  • List<T>内部维护一个T[]数组。

  • 当容量不足时会 申请更大数组 并 拷贝数据。

  • 对象销毁后,这些数组最终交给GC回收。

PooledList<T>内部
  • 内部数组不是直接new出来的,而是从ArrayPool<T>.Shared租借。

  • 使用结束时通过Dispose()方法 归还数组,供下次复用。

  • 避免频繁分配和回收大数组,降低GC Gen2压力。

基本用法

创建与释放
usingMicrosoft.Toolkit.HighPerformance.Buffers;usingvarlist=newPooledList<int>();// 自动使用 ArrayPool<int>list.Add(1);list.Add(2);list.Add(3);foreach(variteminlist){Console.WriteLine(item);}// Dispose() 会自动归还数组

💡 推荐使用 using 确保 Dispose() 被调用,否则不会归还数组,造成内存浪费。

初始容量
usingvarlist=newPooledList<int>(initialCapacity:1024);
转换为Span<T>/Memory<T>

PooledList<T>的优势之一是可以直接获取底层内存:

Span<int>span=list.AsSpan();Memory<int>memory=list.AsMemory();

这样可以高效地与Span/Memory API交互,避免额外拷贝。

常用操作

List<T>基本一致:

list.Add(10);list.AddRange(new[]{20,30,40});list.Insert(1,15);list.RemoveAt(0);list.Clear();Console.WriteLine(list.Count);Console.WriteLine(list.Capacity);

List<T>对比

特性List<T>PooledList<T>
内存分配每次扩容new新数组ArrayPool<T>租借,复用数组
GC 压力大量频繁创建/销毁时 GC 压力大减少 GC Gen2 压力
释放方式依赖 GC必须Dispose()归还数组
性能(频繁操作场景)可能产生大量堆分配高性能、低分配
Span/Memory 支持需要AsSpan()扩展内置AsSpanAsMemory,零拷贝访问
适用场景通用集合高性能、临时性数据集合(网络、序列化、算法)

高级用法

ArrayPool<T>配合
usingvarlist=newPooledList<byte>(ArrayPool<byte>.Shared,2048);

可以传入自定义池,比如为特殊场景优化的ArrayPool<T>

Span<T>高效处理

适合序列化/反序列化:

Span<byte>buffer=list.AsSpan();ProcessBuffer(buffer);// 直接操作底层数组,无需复制
搜索和扩展
varlist=newPooledList<string>{"apple","banana","cherry"};boolfound=list.TryFind(s=>s.StartsWith("b"),outstringresult);Console.WriteLine(found?result:"Not found");// 输出: bananavarpooledFromEnumerable=Enumerable.Range(1,5).ToPooledList();// 扩展方法Console.WriteLine(string.Join(", ",pooledFromEnumerable));// 输出: 1, 2, 3, 4, 5pooledFromEnumerable.Dispose();
  • TryFindTryFindLast:返回boolout值,避免null检查。

注意事项与最佳实践

必须调用 Dispose()
  • 否则不会归还数组,导致内存泄漏。

  • 推荐using块。

不要长期持有 Span/Memory
  • 因为数组归还池后可能被其他线程复用。
适用场景
  • 高频率、大数据临时集合。

  • 网络协议解析、日志聚合、临时缓存。

  • 需要Span访问的场景。

不适合场景
  • 长期持有的全局集合。

  • 数据量小且生命周期长的普通集合。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/25 8:55:46

MSBuild BuildCheck终极实战指南:从构建问题到质量保证的高效解决方案

MSBuild BuildCheck终极实战指南&#xff1a;从构建问题到质量保证的高效解决方案 【免费下载链接】msbuild msbuild: 是 .NET Framework 的构建引擎&#xff0c;用于构建和管理 .NET 项目。适合 .NET 开发者和系统管理员使用 msbuild 构建和管理 .NET 项目。 项目地址: http…

作者头像 李华
网站建设 2026/5/1 4:58:52

14、Bison解析器:语法规则、符号处理与多语法处理策略

Bison解析器:语法规则、符号处理与多语法处理策略 1. 递归语法与栈大小控制 在某些情况下,一个包含5000条语句的程序可能会被解析为一个包含10000个元素(语句和分号)的列表。对于大多数Bison解析器来说,处理一个包含10000个元素的右递归列表可能会过大。右递归语法适用于…

作者头像 李华
网站建设 2026/4/17 18:56:25

20、CD刻录与命令行使用全解析

CD刻录与命令行使用全解析 1. CD光盘类型及使用 CD光盘主要有CD - R和CD - RW两种类型,它们在使用上有不同的特点,具体如下表所示: |光盘类型|可擦除|CD - R驱动器录制|CD - R驱动器播放|CD - RW驱动器录制|CD - RW驱动器播放|普通CD播放器播放| | ---- | ---- | ---- |…

作者头像 李华
网站建设 2026/4/21 12:42:56

15、Bison解析器:冲突处理与特殊功能详解

Bison解析器:冲突处理与特殊功能详解 1. 纯解析器与y.output文件 纯解析器在多线程程序中非常有用,每个线程可以解析来自不同源的输入。 Bison可以创建一个日志文件,传统上命名为 y.output ,现在更常见的是 name.output ,它显示解析器中的所有状态以及状态之间的转…

作者头像 李华
网站建设 2026/4/23 12:11:35

8、Linux 文件管理全攻略

Linux 文件管理全攻略 在日常的文件操作中,我们常常会遇到各种各样的需求,比如获取文件信息、分割文件、比较文件内容、压缩文件、管理文件存档以及跟踪文件修订等。本文将详细介绍在 Linux 系统中进行这些操作的方法和工具,帮助你更高效地管理文件。 1. 获取文件信息 在…

作者头像 李华
网站建设 2026/4/24 2:48:45

51、版本控制工具全解析:CVS、Subversion、RCS及其他

版本控制工具全解析:CVS、Subversion、RCS及其他 在软件开发和文档管理中,版本控制是一项至关重要的技术,它可以帮助我们跟踪文件的变化、管理不同版本的代码和文档。本文将详细介绍几种常见的版本控制工具,包括CVS、Subversion、RCS,以及一些字处理软件中的版本控制功能…

作者头像 李华