IMemoryCache 必须通过 AddMemoryCache() 注册而非 new,否则缺失生命周期管理、过期策略与内存压力响应;应注入使用 GetOrCreateAsync 避免并发重复计算,注意引用安全与惰性过期机制。
直接 new MemoryCache 实例是错的——它不走依赖注入生命周期管理,且缺少默认过期策略和内存压力响应机制。正确做法是在 Program.cs(.NET 6+)或 Startup.ConfigureServices 中调用 AddMemoryCache():
services.AddMemoryCache(options => { options.SizeLimit = 1024; }); —— 可选设置总大小上限(单位为任意“大小单位”,需配合 ICacheEntry.SetSize() 使用IMemoryCache 即可,无需手动管理实例GetOrCreateAsync 是最安全的写法,尤其适合数据库查库后缓存的场景;而 Get + Set 组合在高并发下可能重复生成值(缓存击穿)。
cache.GetOrCreateAsync("key", async entry => { entry.SlidingExpiration = TimeSpan.FromMinutes(10); return await GetDataFromDb(); })
entry.AbsoluteExpirationRelativeToNow 和 SlidingExpiration 不能同时设,后者优先级低但更常用null,Get 返回 null 无法区分“没命中”和“存了 null”,建议改用 TryGetValue + 包装类型(如 int?)或约定哨兵值IMemoryCache 的过期是惰性的:只有在访问时检查是否过期,或后台线程定期扫描(默认 1 分钟间隔)。这意味着:
options.CompactionPercentage = 0.1 调整扫描频率和清理比例cache.Remove("key") 或 cache.TryGetValue("key", out var v) && cache.Remove("key"),但不要在热路径频繁调用IMemoryCache 存的是引用,不是深拷贝。如果缓存了一个 List 并在外部修改它,所有读取方都会看到变化——这通常不是预期行为。
new List(cachedList) 或用 ToList()
init、get; only),或返回只读包装(AsReadOnly())DateTime.Now 这类动态值;要用就缓存计算结果,或用工厂委托延迟求值Set 和 Get,多看一眼你塞进去的东西到底“活”成什么样。