在 C# 中,内存优化是提升应用性能和稳定性的关键。以下是常见的内存优化方法,结合示例和最佳实践:
一、对象分配与生命周期优化
1. 减少临时对象创建
问题:频繁创建短生命周期对象会导致 GC 压力增大。
优化方法:
使用StringBuilder替代字符串拼接(尤其在循环中)。
复用对象实例(如使用对象池)。
示例(反例):
csharp
string result = "";
for (int i = 0; i < 1000; i++)
{
result += i.ToString(); // 每次拼接创建新字符串
}
优化后:
csharp
StringBuilder sb = new StringBuilder();
for (int i = 0; i < 1000; i++) {
sb.Append(i); // 复用StringBuilder实例
}
string result = sb.ToString();
2. 使用值类型替代引用类型
场景:频繁创建的小型对象(如坐标点、状态标志)。
优化方法:
使用struct替代class。
使用Span/Memory避免内存分配。
示例:
csharp
public readonly struct Point { // 值类型
public double X { get; }
public double Y { get; }
public Point(double x, double y) => (X, Y) = (x, y);
}
3. 避免装箱拆箱
问题:值类型与object之间的转换会产生临时对象。
优化方法:
使用泛型集合(如List)替代非泛型(如ArrayList)。
使用System.ValueTuple替代早期的Tuple类。
示例(反例):
csharp
ArrayList list = new ArrayList();
list.Add(42); // 装箱:int → object
int value = (int)list[0]; // 拆箱:object → int
优化后:
csharp
List<int> list = new List<int>();
list.Add(42); // 直接存储int,无装箱
int value = list[0]; // 无拆箱
二、集合与数组优化
1. 预分配集合容量
问题:动态扩容会导致内存复制和旧数组垃圾。
优化方法:
使用带初始容量的构造函数(如new List(100))。
示例:
csharp
// 预分配足够容量,避免多次扩容
var users = new List<User>(1000);
2. 使用高效集合类型
场景:根据访问模式选择合适的集合。
推荐类型:
Dictionary<TKey, TValue>:快速查找(O (1))。
HashSet:唯一元素集合。
ConcurrentDictionary<TKey, TValue>:线程安全的字典。
示例:
csharp
// 使用字典替代List,提升查找性能
var userMap = new Dictionary<int, User>();
userMap.TryGetValue(userId, out var user); // O(1)时间复杂度
3. 避免大型稀疏数组
问题:大量空元素的数组浪费内存。
优化方法:
使用Dictionary<int, T>或SparseArray(需自定义)。
三、内存管理与垃圾回收
1. 控制对象生命周期
原则:及时释放不再使用的资源。
方法:
使用using语句管理实现IDisposable的对象。
避免静态集合持有大量对象引用。
示例:
csharp
using (var stream = new FileStream("data.txt", FileMode.Open)) {
// 使用stream...
} // 自动调用Dispose()释放资源