MemoryCache 使用不当导致的一个 BUG

MemoryCache 使用不当导致的一个 BUG

Intro

前几天发现代码里的一个 BUG,原因是 MemoryCache 使用不当,可以对于很多人来说可能都知道,但还是想分享记录一下,避免以后写出同样的 BUG

Sample

直接来看下面的示例吧

await using var services = new ServiceCollection().AddMemoryCache().BuildServiceProvider();Console.WriteLine("----- Bad -----");
GetValidValues(5).Dump();
GetValidValues(8).Dump();List<int> GetValidValues(int threhold)
{var memoryCache = services.GetRequiredService<IMemoryCache>();var values = memoryCache.GetOrCreate("test1", entry =>{return Enumerable.Range(1, 10).ToList();});values.RemoveAll(x => x > threhold);return values;
}

上面的 Dump 是一个扩展方法就是把 list 内的元素输出出来,实现如下:

public static void Dump(this List<int> values)
{var value = string.Join(",", values);Console.WriteLine(value);
}

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

Fix

可以看到第二次输出的结果和我们的期望不同,之所以会出现上面的问题是因为 MemoryCache 的对象是直接保存在内存中的对象,缓存不发生变化时每次都是返回同一个对象,如果发生修改后面再获取的就是修改后的状态了,所以正确的做法应该要返回一个新的对象而不是修改原来的对象,一个修改方法如下:

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

修改后的输出结果如下:

More

MemoryCache 背后实际是一个 ConcurrentDictionary,value 是一个带着过期时间的对象 CacheEntry

在不过期,没有发生变化的时候每次返回都是同一个对象,作为缓存对象,应该进行只读操作,不应该修改缓存的对象,如果需要修改则应创建新的对象,而非使用原来的对象。

References

  • https://github.com/dotnet/runtime/blob/main/src/libraries/Microsoft.Extensions.Caching.Memory/src/MemoryCache.cs#L26

  • https://github.com/dotnet/runtime/blob/main/src/libraries/Microsoft.Extensions.Caching.Memory/src/CacheEntry.cs

  • https://github.com/WeihanLi/SamplesInPractice/blob/master/MemoryCacheSample/Program.cs

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/298285.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Llinux 磁盘配额的搭建和常规问题解答

1 磁盘配额的使用限制仅针对整个分区&#xff1a;磁盘配额实际运行时&#xff0c;是针对“整个分区”进行限制的&#xff0c;例如&#xff0c;如果/dev/hda5载入在/home下&#xff0c;那么&#xff0c;在/home下面的所有目 录都会受到限制。只对一般身份用户有效&#xff1a;并…

22岁少年破解史上最严重网络攻击,拯救全球互联网,三个月后却被FBI逮捕

转自&#xff1a;大数据文摘编译&#xff1a;牛婉杨2017年&#xff0c;一位名叫Marcus Hutchins的少年从有史以来最严重的网络攻击事件“WannaCry 勒索病毒”中拯救了互联网。如果你是个geek&#xff0c;那么你对WannaCry这个名字一定不陌生&#xff0c; 这是一种可以自行传播的…

python安装beautifulsoup失败_Win10环境下python36安装BeautifulSoup出现错误的解决办法

说明&#xff1a;win10 64位系统&#xff0c;Python3.6.3Win10环境下安装BeautifulSoup4貌似没有任何问题&#xff0c;但是当使用时就会报错&#xff0c;错误如下&#xff1a;通过报错信息到相应的位置去查看文件try:is_file os.path.exists(possible_filename)except Excepti…

使用 Git Extensions 简单入门 Git

使用 Git Extensions 简单入门 Git—— 独立观察员 2015.11.25前言关于这个主题&#xff0c;之前我录了段视频教程&#xff0c;在本地看清晰度还可以&#xff0c;但传到优酷上就很不清晰了&#xff0c;即使是后来重制后还是一样不清晰&#xff0c;所以现在想整理成文字版。当然…

各省地图都像些什么?

全世界只有3.14 % 的人关注了爆炸吧知识人们常说中国地图的形状像一只雄鸡&#xff0c;但具体到每一个省区则并没有一个明确的说法。看看下面这一套省区地图对应的有趣的想象&#xff0c;你觉得像还是不像&#xff1f;★安徽有人说安徽像一只斜倒挂着的蝙蝠&#xff0c;你能看出…

android 蓝牙各种UUID(转载)

android 蓝牙各种UUID ServiceDiscoveryServerServiceClassID_UUID {00001000-0000-1000-8000-00805F9B34FB} BrowseGroupDescriptorServiceClassID_UUID {00001001-0000-1000-8000-00805F9B34FB} PublicBrowseGroupServiceClass_UUID {00001002-0000-1000-8000-00805F9B34F…

Silverlight中摄像头的运用—part2

Silverlight 4 中摄像头的运用—part1将跟踪颜色视作输入 好了&#xff0c;我们能够跟踪到这个颜色了&#xff0c;那这么做的意义是什么呢&#xff1f;实际上&#xff0c;我们可以根据它的位置来移动东西。接下来的例子中&#xff0c;创建的一个球会跟随这个颜色一起移动。你可…

python静态变量和静态方法_python的静态成员变量、实例成员变量、静态方法、类方法、实例方法...

标签&#xff1a;静态成员变量(类变量)和普通成员变量(实例变量)静态成员变量只能通过类名.变量名获得&#xff0c;实例成员变量&#xff0c;通过该实例引用.变量名获得。在实例对静态成员变量赋值时&#xff0c;实例python是动态类型的语言&#xff0c;没有特别的标志区分静态…

同学,解决下这个 Bug!

一些解决 Bug 的小技巧大家好&#xff0c;我是鱼皮。学编程的过程中&#xff0c;我们会遇到各式各样的 Bug&#xff0c;也常常因为它们而感到头秃。但随着你不断解决 Bug、积累经验&#xff0c;就会发现其实解决 Bug 也是有套路的。今天分享下鱼皮自己总结的解决 Bug 套路&…

C#编写串口通信程序(转)

一&#xff0e;概述 在Visual Studio 6.0中编写串口通讯程序&#xff0c;一般都使用Microsoft Communication Control&#xff08;简称MSComm&#xff09;的通讯控件&#xff0c;只要通 过对此控件的属性和事件进行相应编程操作&#xff0c;就可以轻松地实现串口通讯。但在Micr…

pfsense下的流量管理(转)

http://www.pppei.net/blog/post/331 在作流量管理时&#xff0c;这些概念很重要&#xff0c;不要迷失。。 这里再对Limiter 的源地址和目的地址做个说明&#xff0c;因为limiter是被应用在Lan接口的Rule里&#xff0c;相对pfsense来说&#xff0c;用户发往 Lan口的流量为In&am…

如果给你一个亿,你想去干嘛?各专业的科研狗是这样回答的……

全世界只有3.14 % 的人关注了爆炸吧知识“如果现在给你一个亿&#xff0c;你想去做什么&#xff1f;”每天都被穷醒的小天一看到&#xff0c;立马展开了丰富的想象力&#xff1a;首先&#xff0c;当然少不了买买买&#xff0c;将一切之前想要的、不想要的&#xff0c;贵的、更贵…

Apache并发处理模块

apache 2.x版本目前有好几种并发处理模块, 需要在编译的时候通过–with-mpmxxx指定&#xff0c;常用的并发处理模式是prefork和worker。prefork这种模式比较古老&#xff0c;纯进程并发&#xff0c;没有线程&#xff0c;处理方式跟apache1.x是一样的&#xff0c;适用于那些没有…

头文件定义全局变量_5.2 C++局部变量与全局变量 | 输出局部全局变量

C局部变量C局部变量是指&#xff1a;在一个函数内部定义的变量&#xff0c;它只在本函数范围内有效&#xff0c;也就是说只有在本函数内才能使用它&#xff0c;在此函数以外是不能使用这些变量的。同样&#xff0c;在复合语句中定义的变量只在本复合语句范围内有效&#xff0c;…

java float转换成long_在Java中如何将float转换为long或int数据类型?

float f1.2f;double dDouble.parseDouble(String.valueOf(f));System.out.println(d);直接转换会设计到精度问题&#xff0c;所以需要借助字符串 保证不丢失数据www.shufadashi.com防采集。楼主您好&#xff0c;如果float是个整数&#xff0c;强制转换即可&#xff0c;可以捕捉…

Spark 1.2 发布,开源集群计算系统

2019独角兽企业重金招聘Python工程师标准>>> Spark 1.2 发布&#xff0c;此版本包括 172 位贡献者和超过 1000 个 commits。 此版本包括 Spark 核心操作和性能改进&#xff1b;添加新的网络传输子系统&#xff0c;进行了较大的改进&#xff1b;Spark SQL 引入了一个…

[转载].SSRAM、SDRAM和Flash简要介绍

转CalmBright兄的博文&#xff1a;http://www.cnblogs.com/CalmBright/archive/2009/07/19/1526569.html Abstract 在用NIos II 调试sdram遇到了其容量计算的问题&#xff0c;现介绍如下 Introduction 问题1&#xff1a;什么是DRAM、SRAM、SDRAM&#xff1f; 答&#xff1a;名词…

调整 Docker 中 nginx 的日志级别

调整 Docker 中 nginx 的日志级别Intro最近发现我们的一个应用产生了很多日志&#xff0c;而这些日志大多都是 nginx 的 access_log&#xff0c;我们默认会把标准输出收集到 es 里分析应用日志&#xff0c;但是很多都是 access_log 就可能会掩盖掉真实的错误日志&#xff0c;所…

Oracle B-Tree Index 原理

一. B-Tree Index 原理 官网说明&#xff1a; No index structure can satisfy all needs, but the self-balancing B-tree index comes closest to optimizing the performance of searches on large sets of data. Each B-tree node holds multiple keys and pointers. The m…