TIOBE 公布的数据显示,C# 以 2.94% 的年度最大涨幅再度获评2025 年度编程语言,这是 C# 在三年内第二次获此殊荣,凭借的是其在榜单上最大的年度增长幅度。
其实,C# 早就不是当年那个贴着"Windows 独占"和"闭源"标签的语言了。从早期的模仿者到如今现代语言特性的引领者,C# 成功完成了向跨平台和开源的巨大转型。

在企业级开发领域,C# 与 Java 的竞争持续了二十多年。相比于 Java 略显冗长和样板化的代码风格,C# 在语法设计的灵活性和演进速度上一直保持着敏锐的嗅觉。对于开发者而言,C# 版本的快速迭代意味着手里有了更多高效的工具。

以下盘点7个在实际开发中非常实用的C#代码技巧,涵盖并发处理、内存优化及新语法特性。
多线程场景下的字典安全策略
在多线程环境操作字典时,手动加锁(lock)往往容易出错且效率不高。不要重复造轮子,ConcurrentDictionary<TKey, TValue> 是专为并发读写设计的结构,内部实现了细粒度的锁机制和原子操作。
低效:
c#
var cache = new Dictionary<string, int>();
// 多线程并发写入时,普通字典非线程安全,可能导致异常或数据覆盖
await Task.WhenAll(dataItems.Select(async item =>
{
// 即使加锁,性能也会受影响
cache[item.Key] = await ProcessItemAsync(item);
}));
推荐:
c#
var cache = new ConcurrentDictionary<string, int>();
// 内部已处理并发控制,读写更高效
await Task.WhenAll(dataItems.Select(async item =>
{
cache[item.Key] = await ProcessItemAsync(item);
}));
避免频繁分配空集合
在返回空数组或列表时,习惯性地 new 一个对象会造成不必要的内存分配。特别是在高频调用的循环或 LINQ 查询中,这会显著增加垃圾回收(GC)的压力。.NET 为此提供了缓存的单例空对象。
低效:
c#
return new string[0]; // 每次都在堆上分配新对象
推荐:
c#
return Array.Empty<string>(); // 使用全局缓存的空实例
同理,在 LINQ 场景下应使用 Enumerable.Empty<T>()。
善用空合并赋值运算符 (??=)
??= 运算符能让空检查和初始化逻辑变得极其简洁。它不仅减少了样板代码,还消除了不必要的嵌套,非常适合用于属性的延迟加载(Lazy Initialization)。
低效:
c#
if (userSettings == null)
{
userSettings = new List<string>();
}
推荐:
c#
// 仅当 userSettings 为 null 时才进行赋值
userSettings ??= new List<string>();
优化日志记录中的字符串开销
C# 10 引入了针对插值字符串的底层优化。在记录日志时,如果直接使用 $ 进行拼接,即便日志级别未开启,字符串拼接的开销依然存在。使用结构化日志参数写法,可以在日志级别关闭时完全跳过插值计算。
有隐形开销:
c#
// 即使 LogLevel 关闭,字符串插值依然会执行,消耗 CPU
_logger.LogInformation($"Order {orderId} processed at {DateTime.Now}");
推荐:
c#
// 模板化参数,只有日志确实需要记录时,才会处理参数
_logger.LogInformation("Order {OrderId} processed at {ProcessTime}", orderId, DateTime.Now);
并行任务优先使用 Task.WhenAll
在异步方法中,如果多个任务之间没有依赖关系,顺次 await 会导致它们串行执行,浪费了并发优势。应当同时启动所有任务,并等待它们全部完成。
低效:
c#
await UploadLogsAsync();
await UpdateDatabaseAsync();
await NotifyUserAsync();
高效:
c#
await Task.WhenAll(
UploadLogsAsync(),
UpdateDatabaseAsync(),
NotifyUserAsync()
);
注:使用 Task.WhenAll 时,若发生异常,抛出的是 AggregateException ,需注意捕获处理。
预设字典容量以避免重哈希
Dictionary 在元素数量超出当前容量时,会触发扩容(Resize)和重哈希(Rehash),这是非常昂贵的操作。如果能预估数据量级,在构造时指定容量可以大幅减少内存分配开销。
低效:
c#
var map = new Dictionary<int, string>(); // 默认容量小,数据多时会多次扩容
高效:
c#
var map = new Dictionary<int, string>(expectedCount); // 一次分配到位
原始字符串字面量与插值
C# 11 引入了三重引号 """,完美解决了 JSON、SQL 或 HTML 字符串中引号转义的痛点。结合 $$ 语法,还可以自定义插值符号,避免与内容中的 {} 冲突。
c#
var userName = "Alice";
var userAge = 28;
// 使用 $$ 前缀,表示 {{}} 才是插值,单个 {} 被视为普通字符
var jsonContent = $$"""
{
"user": "{{userName}}",
"age": {{userAge}},
"role": "admin"
}
""";
这种写法让代码清晰度大幅提升,再也不用数反斜杠了。
快速搞定.NET环境
上述技巧跨越了多个 C# 版本,从基础的 .NET Framework 到最新的 .NET Core 及 .NET 5+。在实际工作中,维护旧项目(如 .NET 2.0)和探索新特性(如 .NET 10.0)往往需要同时进行。
在本地配置.NET环境时,多个运行版本还是挺麻烦的,涉及大量的环境变量切换。使用 ServBay 就能一键安装 .NET 环境,支持范围极广,从 .NET 2.0 一直到 .NET 10.0,甚至包含了 Mono 6。

ServBay 支持多个.NET版本同时并存,完全不需要开发者手动处理环境变量冲突。无论你是需要维护十年前的老系统,还是测试最新的 C# 语法特性,都能在同一台机器上无缝切换,将精力真正集中在代码编写上。
结语
代码的整洁与高效往往体现在细节之中。掌握这些 C# 技巧,不仅能减少运行时的资源消耗,更能让代码逻辑清晰易读。随着 .NET 生态的不断发展,保持对新特性的关注,并结合得力的工具来管理开发环境,是每一位开发者持续进阶的关键。