引言
C# 再次登顶!2025 年 TIOBE 年度编程语言榜单揭晓,C# 摘得桂冠。这已经是 C# 三年内第二次获此殊荣。
但语言再好,代码写得慢也是白搭。
你是否遇到过这样的场景:
- 循环处理 10 万条数据,跑了几分钟
- 字符串拼接内存爆炸
- LINQ 查询看似优雅,实则性能堪忧
今天我们就来聊聊 7 个实战技巧,让你的 C# 代码运行时间大幅缩短。
一、选对数据结构:O(n) vs O(1)
错误示范
// 用 List 查找,时间复杂度 O(n)
var list = new List<int> { 1, 2, 3, /* ...10万条 */ };
if (list.Contains(99999)) { /* ... */ }
正确做法
// 用 HashSet 查找,时间复杂度 O(1)
var set = new HashSet<int> { 1, 2, 3, /* ...10万条 */ };
if (set.Contains(99999)) { /* ... */ }
性能对比
| 操作 | List | HashSet | Dictionary |
|---|---|---|---|
| 查找 | O(n) | O(1) | O(1) |
| 插入 | O(1) | O(1) | O(1) |
| 删除 | O(n) | O(1) | O(1) |
经验法则:
- 频繁查找 → HashSet / Dictionary
- 需要索引访问 → List
- 需要排序 → SortedSet / SortedDictionary
二、字符串拼接:告别 String,拥抱 StringBuilder
为什么 String 慢?
String 是不可变对象。每次拼接都会创建新对象:
string s = "a";
s += "b"; // 创建新对象 "ab",旧对象 "a" 等待 GC
s += "c"; // 创建新对象 "abc",旧对象 "ab" 等待 GC
循环拼接 10000 次?恭喜,你创建了 10000 个字符串对象。
StringBuilder 登场
var sb = new StringBuilder();
for (int i = 0; i < 10000; i++)
{
sb.Append(i);
}
string result = sb.ToString();
性能对比
// 测试:拼接 10000 次字符串
// String: 约 150ms
string s = "";
for (int i = 0; i < 10000; i++) s += i;
// StringBuilder: 约 1ms
var sb = new StringBuilder();
for (int i = 0; i < 10000; i++) sb.Append(i);
150 倍性能差距,这就是选对工具的力量。
三、Span:零拷贝内存操作
什么是 Span?
Span
- 指向连续内存区域
- 无需复制数据
- 栈分配,无 GC 压力
传统方式 vs Span
// 传统方式:截取子字符串会分配新内存
string text = "Hello, World!";
string sub = text.Substring(0, 5); // 新分配 "Hello"
// Span 方式:零分配
ReadOnlySpan<char> span = text.AsSpan();
ReadOnlySpan<char> subSpan = span.Slice(0, 5); // 无分配
实战案例:解析数字
// 传统方式
string numStr = "12345";
int num = int.Parse(numStr); // 已分配的字符串
// Span 方式
ReadOnlySpan<char> span = "12345".AsSpan();
int num = int.Parse(span); // 直接解析,无需中间字符串
适用场景
- 高频字符串处理
- 大数组切片操作
- 网络数据包解析
- 游戏开发中的缓冲区操作
四、异步编程:不要阻塞线程
错误示范:同步阻塞
// 阻塞主线程
var result = httpClient.GetStringAsync(url).Result;
这会导致:
- 线程池饥饿
- 应用响应变慢
- 死锁风险
正确做法:async/await
// 非阻塞,释放线程
public async Task<string> GetDataAsync(string url)
{
return await httpClient.GetStringAsync(url);
}
并行处理多个任务
// 串行:总时间 = sum(每个任务时间)
var data1 = await GetDataAsync(url1);
var data2 = await GetDataAsync(url2);
var data3 = await GetDataAsync(url3);
// 并行:总时间 = max(每个任务时间)
var tasks = new[]
{
GetDataAsync(url1),
GetDataAsync(url2),
GetDataAsync(url3)
};
var results = await Task.WhenAll(tasks);
3 个请求各 1 秒:
- 串行:3 秒
- 并行:1 秒
五、LINQ 陷阱:延迟执行 vs 立即执行
延迟执行的坑
var numbers = new List<int> { 1, 2, 3 };
var query = numbers.Where(n => n > 1); // 此时未执行
numbers.Add(4); // 修改源数据
var result = query.ToList(); // 这里才执行!
// result = [2, 3, 4],包含了后添加的 4
多次枚举的性能问题
var query = numbers.Where(n => n > 1);
// 错误:每次都重新执行
if (query.Any()) // 执行一次
{
var first = query.First(); // 又执行一次
var count = query.Count(); // 再执行一次
}
// 正确:缓存结果
var list = numbers.Where(n => n > 1).ToList();
if (list.Any())
{
var first = list.First(); // 从缓存读取
var count = list.Count(); // 从缓存读取
}
性能对比
| 操作 | 延迟执行 | 立即执行 (ToList) |
|---|---|---|
| 多次访问 | 每次重新计算 | 一次计算,多次读取 |
| 内存占用 | 低 | 需要存储结果 |
| 适用场景 | 单次遍历 | 多次访问 |
六、避免装箱拆箱
什么是装箱拆箱?
int value = 42;
object obj = value; // 装箱:值类型 → 引用类型
int back = (int)obj; // 拆箱:引用类型 → 值类型
每次装箱都会:
- 在堆上分配内存
- 触发 GC
错误示范
var list = new ArrayList(); // 非泛型集合
list.Add(1); // 装箱
list.Add(2); // 装箱
int sum = 0;
foreach (int i in list) // 拆箱
{
sum += i;
}
正确做法
var list = new List<int>(); // 泛型集合
list.Add(1); // 无装箱
list.Add(2); // 无装箱
int sum = 0;
foreach (int i in list) // 无拆箱
{
sum += i;
}
性能差异
循环 100 万次:
- ArrayList:约 50ms
- List
:约 5ms
10 倍差距,这就是泛型的威力。
七、对象池:复用而非重建
为什么需要对象池?
频繁创建销毁对象会:
- 增加 GC 压力
- 触发频繁 GC 暂停
- 影响应用响应时间
使用 ArrayPool
using System.Buffers;
// 传统方式:每次分配新数组
byte[] buffer = new byte[1024];
ProcessData(buffer);
// 对象池方式:复用数组
var pool = ArrayPool<byte>.Shared;
byte[] buffer = pool.Rent(1024);
try
{
ProcessData(buffer);
}
finally
{
pool.Return(buffer); // 归还池中
}
使用 MemoryPool
using System.Buffers;
var pool = MemoryPool<byte>.Shared;
using var memoryOwner = pool.Rent(1024);
ProcessData(memoryOwner.Memory);
// 自动归还
适用场景
- 高频临时数组
- 网络缓冲区
- 游戏中的临时对象
- 图像处理中的像素缓冲
总结:7 个优化清单
| 技巧 | 收益 | 适用场景 |
|---|---|---|
| 选对数据结构 | O(n) → O(1) | 频繁查找 |
| StringBuilder | 150x 提升 | 字符串拼接 |
| Span |
零拷贝 | 内存操作 |
| async/await | 并行加速 | I/O 操作 |
| LINQ ToList | 避免重复计算 | 多次访问 |
| 泛型集合 | 10x 提升 | 避免装箱 |
| 对象池 | 减少 GC | 高频创建 |
写在最后
性能优化不是一蹴而就的,而是点滴积累。
记住三个原则:
- 先测量,再优化 —— 用 BenchmarkDotNet 找到真正的瓶颈
- 选对工具 —— 数据结构、API、设计模式
- 避免过早优化 —— 可读性优先,性能瓶颈出现时再优化
希望这 7 个技巧能帮到你。我是码农刚子,一个写了六年代码的.NET老程序员。下次见!
