0

0

C# 只读成员(readonly members) - 在结构体中保证不变性

煙雲

煙雲

发布时间:2026-01-23 11:31:32

|

341人浏览过

|

来源于php中文网

原创

readonly成员是c# 7.2引入的结构体不变性保障机制,强制约束readonly实例上调用的方法不修改状态,需显式标注于方法、属性getter等成员,并配合in参数和readonly字段使用。

c# 只读成员(readonly members) - 在结构体中保证不变性

在 C# 中,readonly 成员(C# 7.2 引入)是结构体(struct)实现真正不变性的关键机制。它让编译器强制保证:只要一个结构体实例被标记为 readonly(比如作为只读字段、in 参数或 readonly 局部变量),那么调用其 readonly 成员时,不会意外修改其状态。

为什么结构体需要 readonly 成员?

结构体是值类型,按值传递。但若结构体包含可变字段,又没加约束,很容易在看似“只读”的上下文中被意外修改——比如调用一个非 readonly 的方法,该方法内部却修改了字段。编译器无法阻止这种行为,除非你显式声明成员为 readonly

没有 readonly 成员时:

  • 即使结构体变量被声明为 readonly,仍可调用非 readonly 方法;
  • 这些方法可能悄悄改写字段,破坏语义上的“只读”承诺;
  • 尤其在 in 参数场景下,容易引发难以调试的副作用。

如何正确声明 readonly 成员?

在方法、属性访问器、索引器、运算符等成员上添加 readonly 修饰符:

Jaaz
Jaaz

开源的AI设计智能体

下载
  • 方法:在返回类型前加 readonly,如 public readonly int GetValue() => _value;
  • 属性 getter:在 get 前加 readonly,如 public readonly int Value => _value;
  • 整个属性设为 readonly:表示 get/set 都不能修改状态(set 必须也是 readonly,且通常只用于初始化逻辑);
  • readonly 成员内不可调用非 readonly 成员,也不可直接赋值给任何字段(包括 this 字段)。

readonly 成员与 readonly struct 的区别

readonly struct(C# 7.2+)是更严格的契约:它要求结构体所有实例字段必须是 readonly,且所有成员默认隐式为 readonly(除非显式去掉)。而普通结构体中使用 readonly 成员,是“按需加锁”,更灵活:

  • 允许结构体保留可变字段(如用于内部缓存),但把公共 API 设为 readonly
  • 适合渐进式改造旧结构体,无需一次性冻结所有字段;
  • 编译器对 readonly struct 有额外优化(如允许安全地按 in 传递)。

实际使用建议

要真正发挥 readonly 成员的作用,需配合使用场景:

  • 把结构体字段尽量设为 readonly,从源头减少可变性;
  • 所有公开的、不改变状态的方法/属性,一律标注 readonly
  • 在参数中优先用 in 替代 ref 或值传递,此时编译器会强制只调用 readonly 成员;
  • 避免在 readonly 成员里调用非 readonly 方法(编译器报错),也别用 Unsafe.AsRef 等绕过检查——那等于放弃保障。

基本上就这些。readonly 成员不是语法糖,而是编译器帮你守住结构体不变性的第一道防线。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
java基础知识汇总
java基础知识汇总

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

1564

2023.10.24

Go语言中的运算符有哪些
Go语言中的运算符有哪些

Go语言中的运算符有:1、加法运算符;2、减法运算符;3、乘法运算符;4、除法运算符;5、取余运算符;6、比较运算符;7、位运算符;8、按位与运算符;9、按位或运算符;10、按位异或运算符等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

241

2024.02.23

php三元运算符用法
php三元运算符用法

本专题整合了php三元运算符相关教程,阅读专题下面的文章了解更多详细内容。

148

2025.10.17

golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

490

2025.06.09

golang结构体方法
golang结构体方法

本专题整合了golang结构体相关内容,请阅读专题下面的文章了解更多。

202

2025.07.04

string转int
string转int

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

1010

2023.08.02

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

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

608

2024.08.29

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

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

314

2025.08.29

Go高并发任务调度与Goroutine池化实践
Go高并发任务调度与Goroutine池化实践

本专题围绕 Go 语言在高并发任务处理场景中的实践展开,系统讲解 Goroutine 调度模型、Channel 通信机制以及并发控制策略。内容包括任务队列设计、Goroutine 池化管理、资源限制控制以及并发任务的性能优化方法。通过实际案例演示,帮助开发者构建稳定高效的 Go 并发任务处理系统,提高系统在高负载环境下的处理能力与稳定性。

4

2026.03.10

热门下载

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

精品课程

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

共94课时 | 11.1万人学习

C 教程
C 教程

共75课时 | 5.3万人学习

C++教程
C++教程

共115课时 | 21.3万人学习

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

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