C# 9引入的record类型是一种具有值语义的引用类型,最显著的特征就是默认具备不变性,这一特性在多线程编程场景中能发挥重要作用,有效降低并发数据操作的风险。

record类型的不变性实现
record类型默认的属性都是只读的,初始化之后无法修改属性值,这是其不变性的核心体现。我们可以通过以下简单代码理解这一特性:
// 定义一个record类型
public record UserRecord(int Id, string Name, int Age);
class Program
{
static void Main()
{
// 初始化record实例
UserRecord user = new UserRecord(1, "张三", 25);
// 尝试修改属性会直接编译报错
// user.Name = "李四"; // 这行代码无法通过编译
Console.WriteLine($"用户ID:{user.Id},姓名:{user.Name},年龄:{user.Age}");
}
}
如果需要基于现有record实例生成新的实例,只能通过with表达式创建副本,原实例不会被修改:
UserRecord oldUser = new UserRecord(1, "张三", 25);
// 使用with表达式创建新实例,修改部分属性
UserRecord newUser = oldUser with { Name = "李四", Age = 26 };
Console.WriteLine($"原用户姓名:{oldUser.Name}"); // 输出 张三
Console.WriteLine($"新用户姓名:{newUser.Name}"); // 输出 李四
多线程环境下的不变性优势
1. 无需额外的同步锁开销
多线程环境下操作共享数据时,通常需要加锁保证数据一致性,而record类型的不变性使得实例一旦创建就无法修改,多个线程同时读取同一个record实例时不会产生数据竞争问题,不需要额外的锁机制,减少了性能开销。
using System;
using System.Threading;
using System.Threading.Tasks;
public record ProductRecord(int ProductId, string ProductName, decimal Price);
class MultiThreadTest
{
static void Main()
{
// 创建共享的record实例
ProductRecord sharedProduct = new ProductRecord(1001, "笔记本电脑", 5999.99m);
// 启动多个线程同时读取共享实例
Task task1 = Task.Run(() => PrintProductInfo(sharedProduct, "线程1"));
Task task2 = Task.Run(() => PrintProductInfo(sharedProduct, "线程2"));
Task task3 = Task.Run(() => PrintProductInfo(sharedProduct, "线程3"));
Task.WaitAll(task1, task2, task3);
}
static void PrintProductInfo(ProductRecord product, string threadName)
{
// 无需加锁直接读取,不会出现数据不一致问题
Console.WriteLine($"{threadName}读取:商品ID={product.ProductId},名称={product.ProductName},价格={product.Price}");
}
}
2. 避免共享数据被意外修改
在多线程场景中,一个线程修改共享数据可能导致其他线程读取到异常状态,而record类型的只读属性从语法层面禁止了修改操作,从根源上避免了这类问题,降低了并发bug的出现概率。
3. 提升代码可读性和可维护性
record类型的不变性让数据的生命周期和状态变化更清晰,开发者看到record类型就能明确其是不可变的,不需要额外查看代码逻辑确认是否有修改操作,在多线程场景下能更快理解数据流转逻辑,减少维护成本。
record类型在多线程场景的使用注意点
虽然record类型本身是不可变的,但如果其属性包含可变引用类型的字段,仍然可能存在线程安全问题。例如record包含List<string>类型的属性,虽然record的该属性引用不能修改,但List内部的内容可以被修改,这种情况下还是需要额外的同步处理。
using System.Collections.Generic;
// 包含可变引用类型属性的record
public record OrderRecord(int OrderId, List<string> Items);
class Program
{
static void Main()
{
List<string> items = new List<string> { "商品A", "商品B" };
OrderRecord order = new OrderRecord(1, items);
// 虽然order.Items的引用不能修改,但List内容可以修改
order.Items.Add("商品C"); // 这行代码是合法的
Console.WriteLine($"订单商品数量:{order.Items.Count}");
}
}
因此在使用record类型处理多线程场景的复杂数据时,需要确保其包含的所有引用类型属性也都是不可变的,才能完全发挥record的不变性优势。