0

0

C# 只读集合使用方法 C#如何创建IReadOnlyList或ReadOnlyCollection

星降

星降

发布时间:2026-03-16 10:44:03

|

825人浏览过

|

来源于php中文网

原创

ReadOnlyCollection 是零开销只读包装,但非深只读;IReadOnlyList 是接口,用于契约与泛型约束;ToArray() 拷贝数据,Memory<T> 适用于高性能短生命周期场景。

c# 只读集合使用方法 c#如何创建ireadonlylist或readonlycollection

ReadOnlyCollection 是最常用且安全的只读包装方式

直接用 new ReadOnlyCollection<t>(list)</t> 包装一个已存在的 List<t></t>,这是 .NET Framework 2.0 就支持的方案,运行时开销极小,且能真正阻止修改——调用 AddRemove 等方法会抛出 NotSupportedException

注意它不是“深只读”:如果原 List<t></t> 后续被其他代码修改,ReadOnlyCollection<t></t> 的内容也会变(它是引用包装,不是拷贝)。所以确保原始集合不再被意外修改,或者在构造前先做防御性拷贝:

var source = new List<string> { "a", "b" };
var readOnly = new ReadOnlyCollection<string>(new List<string>(source)); // 防御性拷贝
  • 适用于需要向外部暴露只读视图,但内部仍需维护可变集合的场景
  • 兼容所有 .NET 版本(包括 .NET Framework 4.5+ 和 .NET Core / .NET 5+)
  • 不要对 ReadOnlyCollection<t></t> 调用 AsReadOnly() ——那是 List<t></t> 的扩展方法,返回的仍是 ReadOnlyCollection<t></t>,语义重复

IReadOnlyList 多用于接口契约和泛型约束

IReadOnlyList<t></t> 是接口,不能直接 new。它主要用在参数类型、返回类型或泛型约束中,强调“我只要索引访问 + 不修改”的契约。实际返回时,通常还是用 ReadOnlyCollection<t></t> 或数组(T[])来实现它:

public IReadOnlyList<int> GetNumbers() => new ReadOnlyCollection<int>(new List<int> { 1, 2, 3 });

数组也天然实现 IReadOnlyList<t></t>(从 .NET 4.5 开始),所以如果数据固定,直接返回 new[] { 1, 2, 3 } 更轻量。

  • 避免在方法内部 new List<t>().AsReadOnly()</t> 后再转成 IReadOnlyList<t></t> ——这多一次装箱和间接层,不如直接 new ReadOnlyCollection<t></t>
  • 若泛型方法约束为 where T : IReadOnlyList<U>,传入 ReadOnlyCollection<u></u>U[] 都合法;但传 List<u></u> 不行(它不实现该接口)
  • IReadOnlyList<t></t> 没有 Count 属性的 setter,但有 Count getter 和 this[int] 索引器,比 IEnumerable<t></t> 更适合随机访问场景

别误用 Array.AsReadOnly() ——它不存在

数组没有 AsReadOnly() 方法。常见错误是看到 List<t>.AsReadOnly()</t> 就类推到数组,结果编译失败。数组本身就是只读的(除非你用 Array.Copy 或反射强行改),且已实现 IReadOnlyList<t></t>,不需要额外包装。

iMuse.AI
iMuse.AI

iMuse.AI 创意助理,为设计师提供无限灵感!

下载

如果你写 myArray.AsReadOnly(),编译器会报错:'T[]' does not contain a definition for 'AsReadOnly'

  • 正确做法:直接返回数组,或显式转型 (IReadOnlyList<T>)myArray(通常不需要,协变/隐式转换已覆盖)
  • 若真需要“不可变数组副本”,用 Array.AsReadOnly() 是错的;应改用 ReadOnlyCollection<t>(myArray.ToList())</t>,或更高效地用 Memory<T>.ToArray().AsReadOnly()(仅限 .NET Core 2.1+)

性能与语义:ReadOnlyCollection vs ToArray() vs Memory

三者都提供只读语义,但成本和适用场景不同:

  • ReadOnlyCollection<t></t>:零分配(仅包装对象),适合长期持有、频繁索引访问,但依赖原始集合生命周期
  • ToArray():分配新数组,内容拷贝,内存开销大,适合一次性返回、避免外部引用干扰
  • Memory<T>(配合 ReadOnlyMemory<T>):.NET Core 2.1+ 引入,支持栈分配(stackalloc)、无 GC 压力,适合高性能、短生命周期场景(如解析、序列化中间表示);但它不是集合接口,不能当 IReadOnlyList<t></t> 直接用,需通过 .ToArray()Span<T>.ToArray() 转换

最容易被忽略的一点:ReadOnlyCollection<t></t>Count 和索引器都是 O(1),但它的枚举器(GetEnumerator())会委托给底层列表——如果底层列表是自定义实现且 GetEnumerator() 有副作用或性能问题,这个委托就会暴露出来。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

WorkBuddy
WorkBuddy

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
counta和count的区别
counta和count的区别

Count函数用于计算指定范围内数字的个数,而CountA函数用于计算指定范围内非空单元格的个数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

203

2023.11.20

string转int
string转int

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

1071

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

617

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

335

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

235

2025.08.29

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1998

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

681

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

2418

2025.12.29

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

90

2026.03.13

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 11.4万人学习

C 教程
C 教程

共75课时 | 5.5万人学习

C++教程
C++教程

共115课时 | 22.1万人学习

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

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