#yyds干货盘点#分析一个错误使用MemoryCache导致的BUG

Posted 494324190

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了#yyds干货盘点#分析一个错误使用MemoryCache导致的BUG相关的知识,希望对你有一定的参考价值。

这个Bug是我在项目中发现的,原因是MemoryCache使用不当造成了一个不小的Bug,虽说这个Bug很大部分人都知道,但是我觉得还是分享出来,记录一下。废话不多说,我们来看一下出错的代码(代码已经经过脱敏处理)

await using var services =     new ServiceCollection()        .AddMemoryCache()        .BuildServiceProvider();GetValidValues(5).Dump();GetValidValues(8).Dump();List<int> GetValidValues(int valueInt){    var memoryCache = services.GetRequiredService<IMemoryCache>();    var vs= memoryCache.GetOrCreate("t1", entry =>    {        return Enumerable.Range(1, 10).ToList();    });    vs.RemoveAll(x => x > valueInt);    return vs;}

代码中Dump是扩展方法,它是把list内的元素输出出来,具体实现代码如下:

public static void Dump(this List<int> vs){    string v= string.Join("--", vs);    Console.WriteLine(v);}

好了,来想一下上面的输出结果会是什么吧,期望的结果应该是每次都输出小于等于输入的值,实际是什么样的呢?实际输出结果如下:

#yyds干货盘点#分析一个错误使用MemoryCache导致的BUG_缓存

从上图中第二次输出的结果是不是和你想的不一样呢,之所以出现上面问题是因为MemoryCache对象是直接保存在内存中的,缓存不变化时每次都返回同一个对象,如果发生了修改那么再次获取就是修改后的内容。因此正确做法是返回一个新对象而不是修改原来的对象,一个修改方法如下:

List<int> GetValidValues(int valueInt){    var memoryCache = services.GetRequiredService<IMemoryCache>();    var vs= memoryCache.GetOrCreate("t1", entry =>    {        return Enumerable.Range(1, 10).ToList();    });    return vs.Where(v => v <= valueInt).ToList();}

修改后的输出结果如下:

#yyds干货盘点#分析一个错误使用MemoryCache导致的BUG_扩展方法_02

总结:

MemoryCache背后其实就是ConcurrentDictionary,value其实是带着过期时间的CacheEntry,因此在不过期并且没有发生变化的时候每次返回都是同一个缓存对象。作为缓存对象应进行只读操作,不应修改缓存对象,如需要修改应创建新对象而不是使用原来的对象。

以上是关于#yyds干货盘点#分析一个错误使用MemoryCache导致的BUG的主要内容,如果未能解决你的问题,请参考以下文章

JavaScript之预编译学习(内含多个面试题) #yyds干货盘点#

#yyds干货盘点# Java 内存分析之堆内存和MetaSpace内存

#yyds干货盘点#盘点MySQL中的那些日志文件

#yyds干货盘点# 性能问题分析策略

从JVM堆内存分析验证深浅拷贝#yyds干货盘点#

#yyds干货盘点# 性能问题分析策略