浅谈MemoryCache的原生插值方式

news/2024/7/24 8:13:08 标签: java, c#, redis, 缓存, .net

b155aa674e9f52fefaf7c0bbf10f29f4.gif

.NET运行时内置了常用的缓存模块:MemoryCache

标准的MemoryCache暴露了如下几个属性和方法:

public int Count { get; }
public void Compact(double percentage);
public ICacheEntry CreateEntry(object key);
public void Dispose();
public void Remove(object key);
public bool TryGetValue(object key, out object result);
protected virtual void Dispose(bool disposing);

但是你使用常规模式去插值/获取值,可能会出现意想不到的情况。

就如下这样的常规代码:

var s = new MemoryCache(new MemoryCacheOptions { });
var entry = s.CreateEntry("WeChatID");
entry.Value = "精益码农";

var f =  s.TryGetValue("WeChatID",out  object obj);

Console.WriteLine(f);
Console.WriteLine(obj);

会输出如下结果:b4a2a7ccc4a2903bb9091a081c5eef9a.png

是不是很意外。


但是看官们一般不会使用MemoryCache的原生方法,而是使用位于同一命名空间的 扩展方法Set

var s = new MemoryCache(new MemoryCacheOptions { });
s.Set("WeChatID", "精益码农");
var f = s.TryGetValue("WeChatID", out object obj);

Console.WriteLine(f);
Console.WriteLine(obj);

如此便能正确输出。40aa13c56fc21db2e7a3c095d78260bf.png

扩展类源码看一看

public static TItem Set<TItem>(this IMemoryCache cache, object key, TItem value)
 {
      using ICacheEntry entry = cache.CreateEntry(key);
      entry.Value = value;
      return value;
}

扩展方法与原生方法的差异在于using关键字 (也说明了CacheEntry继承自IDisposable接口)。

继续追溯CacheEntry实现的Dispose方法:

public void Dispose()
        {
            if (!_state.IsDisposed)
            {
                _state.IsDisposed = true;

                if (_cache.TrackLinkedCacheEntries)
                {
                    CacheEntryHelper.ExitScope(this, _previous);
                }

                // Don't commit or propagate options if the CacheEntry Value was never set.
                // We assume an exception occurred causing the caller to not set the Value successfully,
                // so don't use this entry.
                if (_state.IsValueSet)
                {
                    _cache.SetEntry(this);

                    if (_previous != null && CanPropagateOptions())
                    {
                        PropagateOptions(_previous);
                    }
                }

                _previous = null; // we don't want to root unnecessary objects
            }
        }

注意其中的_cache.SetEntry(this),表示在MemoryCache底层的ConcurrentDictionary<object, CacheEntry>集合插入缓存项,

综上:缓存项CacheEntry需要被Dispose,才能被插入MemoeyCache

这是怎样的设计模式?IDisposable接口不是用来释放资源吗?
为啥要使用Dispose方法来向MemoryCache插值?
不能使用一个明确的Commit方法吗?

这在Github上也有issue讨论,从2017年开始就有大佬质疑这是一个反人类的设计思路,官方为了不引入Break Change,一直保持到现在。


基于此现状,我们如果使用MemoryCache的原生插值方法, 需要这样:

var s = new MemoryCache(new MemoryCacheOptions { });
 using (var entry = s.CreateEntry("WeChatID"))
 {
      entry.Value = "精益码农";
 }
 var f = s.TryGetValue("WeChatID", out object obj);
 ...

尽量不要使用C#8.0推出的不带大括号的using语法

using var entry = s.CreateEntry("WeChatID");
 entry.Value = "精益码农";
            
 var f = s.TryGetValue("WeChatID", out object obj);
 ...

这种没明确指定using作用范围的语法,会在函数末尾才执行Dispose方法, 导致执行到TryGetValue时,缓存项其实还没插入!!!

Last
  1. MemoryCache插值的实现过程很奇葩

  2. 尽量使用带明确大括号范围的using语法,C#8.0推出的不带大括号的using语法糖的作用时刻在函数末尾,会带来误导。

●HTTP1.1 Keep-Alive到底算不算长连接?

●宝藏好物gRPCurl

●SignalR 开发到生产部署闭坑指南

●SignalR在React/Go技术栈的实践

●我是状态机, 一颗永远骚动的机器引擎

●大揭秘| 我司项目组Gitlab Flow && DevOps流程

●你怕是对MD5算法有误解

4145d5c65926f80444ce328e527ac7e6.png

点个在看你最好看

6c8ea9fea0e47fa1e896dbbcde4ccf0b.png


http://www.niftyadmin.cn/n/1864454.html

相关文章

sql得到时间

sql得到当前系统时间得 日期部分 CONVERT(varchar(10),getDate(),120) 昨天 select convert(varchar(10),getdate() - 1,120) 明天 select convert(varchar(10),getdate() 1,120) 最近七天 select * from tb where 时间字段 > convert(varchar(10),getdate() - 7,120) 随后…

[导入]从架构设计到系统实施——基于.NET 3.0的全新企业应用系列课程(5):设计基于WPF的客户端.zip(6.98 MB)...

讲座内容&#xff1a; 本次系列课程将采用案例教学的方法&#xff0c;深度剖析微软的基于.NET 3.0的分布式应用设计方法和实施过程。在第五次课程中&#xff0c;我们将详细介绍如何设计基于.NET 3.0 WPF的全新的企业应用的具有相当好的客户体验的客户端。 课程讲师&#xff1a;…

一条nginx命令引发的对于容器的思考

去年的时候写了一篇原创《前后端分离&#xff0c;如何在前端项目中动态插入后端API基地址&#xff1f;&#xff08;in docker&#xff09;》&#xff0c; 我自认为这篇生产实践是对大前端、 容器化、CI/CD的得意之作。对于前后端分离的web项目&#xff0c;在容器启动的瞬间&…

vs 2008 不能切换到设计视图的解决办法

vs 2008 不能切换到设计视图的解决办法 家里电脑上安装了vs2008,之前没装其他版本的vs,最近在使用webform窗体的时候发现不能切换到设计视图了,一点切换vs就卡死,网上搜索了一下 除了重装vs2008 没有发现其他解决办法,发现vs2005也有这个问题,vs2005的解决办法是http://www.cnb…

2021技术文大盘点 | 打包过去,​面向未来

先用四句诗词快速描述 一下我的写作心得1. 只在此山中&#xff0c;云深不知处作为开发人员&#xff0c;常执着于机器0,1代码&#xff0c;非假既真&#xff1b;真实世界是很主观的&#xff0c;需要精致细节&#xff0c;更多时候需要全局把控。带着问题写作&#xff0c;对事物理解…

PInvoke

1&#xff0c; PInvoke什么意思&#xff1f; Platform Invocation Services2, 干什么用? 导入外部函数&#xff1f;什么是外部函数&#xff0c;就是不属于.Net托管的函数。3&#xff0c;如何用&#xff1f;看下面的例子。用[DllImport(dllname)]来实现&#xff0c;但是首先要…

排查go开发的HttpClient读取Body超时

记一次go httpclient [读取响应Body超时]的排查过程。今年度解锁的第一个技能。01故障现场本人负责的主备集群,发出的 HttpClient 请求有 30%概率超时&#xff0c; 报context deadline exceeded (Client.Timeout or context cancellation while reading body) 异常Kibana 显示 …

基于JQUERY制作的仿GOOGLE自动完成插件

基于JQUERY制作的仿GOOGLE自动完成插件 数据采用JSON&#xff0c;格式为{keylist:[{keyname:关键字1,keyextend:扩展文字(譬如说结果数目)},{keyname:关键字2,keyextend:扩展文字(譬如说结果数目)}]} 效果如图 完整的JS代码如下 Code1/**//* 2 * jQuery AutoComplete 3 * 4 …