一文读懂NetCore的缓存
关于我
缓存基础知识
缓存可以减少生成内容所需的工作,从而显著提高应用程序的性能和可伸缩性。 缓存最适用于不经常更改的 数据,生成 成本很高。 通过缓存,可以比从数据源返回的数据的副本速度快得多。 应该对应用进行编写和测试,使其 永不 依赖于缓存的数据。
ASP.NET Core 支持多个不同的缓存。 最简单的缓存基于 IMemoryCache。 IMemoryCache
表示存储在 web 服务器的内存中的缓存。 在服务器场上运行的应用 (多台服务器) 应确保会话在使用内存中缓存时处于粘滞状态。 粘滞会话确保来自客户端的后续请求都将发送到相同的服务器。
内存中缓存可以存储任何对象。 分布式缓存接口仅限 byte[]
。 内存中和分布式缓存将缓存项作为键值对。
缓存指南
- 代码应始终具有回退选项,以获取数据,而 不是依赖于可用的缓存值。
- 缓存使用稀有资源内存,限制缓存增长:
- 不要 使用外部 输入作为缓存键。
- 使用过期限制缓存增长。
- 使用 SetSize、Size 和 SizeLimit 限制缓存大小]。 ASP.NET Core 运行时不会根据内存 压力限制缓存 大小。 开发人员需要限制缓存大小。
使用
DI注入
创建一个NetCore控制台项目,进行缓存的项目演示。
控制台项目只有一个初始化的Program.cs文件。基于NetCore进行项目编码,每一步就是创建一个基础模板,使用依赖注入的方式。
nuget install Microsoft.Extensions.Hosting
public static class Program { static async void Main(string[] args) { var builder = new HostBuilder().ConfigureServices((context, service) => { }); await builder.RunConsoleAsync(); } }
注入缓存服务,控制台需要下载库 Microsoft.Extensions.Caching.Memory
nuget install Microsoft.Extensions.Caching.Memory
public static class Program { static async void Main(string[] args) { var builder = new HostBuilder().ConfigureServices((context, service) => { service.AddMemoryCache(); service.AddScoped<CacheService>();//实际测试服务 service.AddHostedService<BackgroundJob>();//后台执行方法 }); await builder.RunConsoleAsync(); } }
后台服务
public class BackgroundJob : IHostedService { private readonly CacheService _cacheService; public BackgroundJob(CacheService cacheService) { _cacheService = cacheService; } public Task StartAsync(CancellationToken cancellationToken) { _cacheService.Action(); return Task.CompletedTask; } public Task StopAsync(CancellationToken cancellationToken) { return Task.CompletedTask; } }
MemoryCache使用总结
通过构造函数自动注入IMemoryCache
public class CacheService { private readonly IMemoryCache _memoryCache; public CacheService(IMemoryCache memoryCache) { _memoryCache = memoryCache; } }
最基本的使用
Set方法根据Key设置缓存,默认缓存不过期
Get方法根据Key取出缓存
/// <summary> /// 缓存设置 /// </summary> public void BaseCache() { string cacheKey = "timestamp"; //set cache _memoryCache.Set(cacheKey, DateTime.Now.ToString()); //get cache Console.WriteLine(_memoryCache.Get(cacheKey)); }
IMemoryCache提供一些好的语法糖供开发者使用,具体内容看下方文档
/// <summary> /// 特殊方法的使用 /// </summary> public void ActionUse() { //场景-如果缓存存在,取出。如果缓存不存在,写入 //原始写法 string cacheKey = "timestamp"; if (_memoryCache.Get(cacheKey) != null) { _memoryCache.Set(cacheKey, DateTime.Now.ToString()); } else { Console.WriteLine(_memoryCache.Get(cacheKey)); } //新写法 var dataCacheValue = _memoryCache.GetOrCreate(cacheKey, entry => { return DateTime.Now.ToString(); }); Console.WriteLine(dataCacheValue); //删除缓存 _memoryCache.Remove(cacheKey); //场景 判断缓存是否存在的同时取出缓存数据 _memoryCache.TryGetValue(cacheKey, out string cacheValue); Console.WriteLine(cacheValue); }
缓存过期策略
设置缓存常用的方式主要是以下二种
- 绝对到期(指定在一个固定的时间点到期)
- 滑动到期(在一个时间长度内没有被命中则过期)
- 组合过期 (绝对过期+滑动过期)
绝对到期
过期策略 5秒后过期
//set absolute cache string cacheKey = "absoluteKey"; _memoryCache.Set(cacheKey, DateTime.Now.ToString(), TimeSpan.FromSeconds(5)); //get absolute cache for (int i = 0; i < 6; i++) { Console.WriteLine(_memoryCache.Get(cacheKey)); Thread.Sleep(1000); }
滑动到期
过期策略 2秒的滑动过期时间,如果2秒内有访问,过期时间延后。当2秒的区间内没有访问,缓存过期
//set slibing cache string cacheSlibingKey = "slibingKey"; MemoryCacheEntryOptions options = new MemoryCacheEntryOptions(); options.SlidingExpiration = TimeSpan.FromSeconds(2); _memoryCache.Set(cacheSlibingKey, DateTime.Now.ToString(), options); //get slibing cache for (int i = 0; i < 2; i++) { Console.WriteLine(_memoryCache.Get(cacheSlibingKey)); Thread.Sleep(1000); } for (int i = 0; i < 2; i++) { Thread.Sleep(2000); Console.WriteLine(_memoryCache.Get(cacheSlibingKey)); }
组合过期
过期策略
6秒绝对过期+2秒滑动过期
满足任意一个缓存都将失效
string cacheCombineKey = "combineKey"; MemoryCacheEntryOptions combineOptions = new MemoryCacheEntryOptions(); combineOptions.SlidingExpiration = TimeSpan.FromSeconds(2); combineOptions.AbsoluteExpiration = DateTime.Now.AddSeconds(6); _memoryCache.Set(cacheCombineKey, DateTime.Now.ToString(), combineOptions); //get slibing cache for (int i = 0; i < 2; i++) { Console.WriteLine(_memoryCache.Get(cacheCombineKey)); Thread.Sleep(1000); } for (int i = 0; i < 6; i++) { Thread.Sleep(2000); Console.WriteLine(i+"|" + _memoryCache.Get(cacheCombineKey)); } Console.WriteLine("------------combineKey End----------------");
缓存状态变化事件
当缓存更新、删除时触发一个回调事件,记录缓存变化的内容。
/// <summary> /// cache状态变化回调 /// </summary> public void CacheStateCallback() { MemoryCacheEntryOptions options = new MemoryCacheEntryOptions(); options.AbsoluteExpiration = DateTime.Now.AddSeconds(3 ); options.RegisterPostEvictionCallback(MyCallback, this); //show callback console string cacheKey = "absoluteKey"; _memoryCache.Set(cacheKey, DateTime.Now.ToString(), options); Thread.Sleep(500); _memoryCache.Set(cacheKey, DateTime.Now.ToString(), options); _memoryCache.Remove(cacheKey); } private static void MyCallback(object key, object value, EvictionReason reason, object state) { var message = $"Cache entry state change:{key} {value} {reason} {state}"; ((CacheService)state)._memoryCache.Set("callbackMessage", message); Console.WriteLine(message); }
缓存依赖策略
设置一个缓存A 设置一个缓存B,依赖于缓存A 如果缓存A失效,缓存B也失效
/// <summary> /// 缓存依赖策略 /// </summary> public void CacheDependencyPolicy() { string DependentCTS = "DependentCTS"; string cacheKeyParent = "CacheKeys.Parent"; string cacheKeyChild = "CacheKeys.Child"; var cts = new CancellationTokenSource(); _memoryCache.Set(DependentCTS, cts); //创建一个cache策略 using (var entry = _memoryCache.CreateEntry(cacheKeyParent)) { //当前key对应的值 entry.Value = "parent" + DateTime.Now; //当前key对应的回调事件 entry.RegisterPostEvictionCallback(MyCallback, this); //基于些key创建一个依赖缓存 _memoryCache.Set(cacheKeyChild, "child" + DateTime.Now, new CancellationChangeToken(cts.Token)); } string ParentCachedTime = _memoryCache.Get<string>(cacheKeyParent); string ChildCachedTime = _memoryCache.Get<string>(cacheKeyChild); string callBackMsg = _memoryCache.Get<string>("callbackMessage"); Console.WriteLine("第一次获取"); Console.WriteLine(ParentCachedTime + "|" + ChildCachedTime + "|" + callBackMsg); //移除parentKey _memoryCache.Get<CancellationTokenSource>(DependentCTS).Cancel(); Thread.Sleep(1000); ParentCachedTime = _memoryCache.Get<string>(cacheKeyParent); ChildCachedTime = _memoryCache.Get<string>(cacheKeyChild); callBackMsg = _memoryCache.Get<string>("callbackMessage"); Console.WriteLine("第二次获取"); Console.WriteLine(ParentCachedTime + "|" + ChildCachedTime + "|" + callBackMsg); }
参考资料
Asp.Net Core 轻松学-在.Net Core 使用缓存和配置依赖策略
拥抱.NET Core系列:MemoryCache 缓存过期
https://www.cnblogs.com/ants/p/8482227.html
https://blog.csdn.net/u010476739/article/details/102947433
推荐阅读
最后
本文到此结束,希望对你有帮助 😃
如果还有什么疑问或者建议,可以多多交流,原创文章,文笔有限,才疏学浅,文中若有不正之处,万望告知。
更多精彩技术文章汇总在我的 公众号【程序员工具集】,持续更新,欢迎关注订阅收藏。

低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
- 上一篇
编码之道(二):软件的价值
程序员最主要的一个工作就是编码,编码只是个过程而已,最终编码的目的就是产生一个能提供服务的有价值的软件。 不管你负责的是后端编码, 产生的是可部署运行的服务,或是移动端构建了一个App,也许是前端,编写了页面等,也许可能是类库或框架等。所有这些产物,如果我们用一个词来归纳它们,那就是软件 那做为程序员,你有没有思考过,软件究竟有什么价值? 本周,继续就编码之道阐述我的思考与分析,这是第二篇,本系列其它文章为: 编码之道(一):程序员的"圣经" 为什么要谈价值 可能有些人觉得谈论软件的价值是有点多余,因为软件一定是有价值的,客户需要一个软件,肯定它能满足客户一定的需求。 所以这个点并无太多谈论的必要。 这也正是我想要写这篇文章的原因,这正是因为软件的利益方,包括客户,程序员,管理人员,公司等各方角色,在识别软件价值上都存在误差。 而这种误差,正在造成现在软件行业困境的一个很重要的原因。 我们软件行业的最大困境就是难以产生高质量易于维护的软件 所以我们程序员,是否能清晰的理解软件的价值,这是我们能写好代码的一个基础。 因为: 软件有看得见的价值与看不见的价值 而在编码中,很多问题的频繁出现...
- 下一篇
文档内容结构化在百度文库的技术探索
导读:简述百度文库关于各类文档的转码和展现历程,早期的版式数据满足了PC端的各类文档阅读体验,随着业务发展的需求迭代,无线端的文档阅读体验亟需提升。版式数据转流式数据过程中,简易的内容结构化满足了pdf数据在无线端的重排版。底层解析ooxml数据和细致的内容结构化,则带来了不错的word无线端重排版效果。从chart图片中“从无到有”抽取结构化的元数据,更为用户与文档的互动打开了想象空间 全文3724字,预计阅读时间 9分钟。 一、百度文库中各类文档的展现 文库有数十亿海量文档,包括word,ppt,excel和pdf等十几种常见办公文档,核心基础服务是文档转码和展现。 为了统一十几种文档的转码和展现方案,不依赖于原文件格式的开档软件,技术调研后,最终方案为任意文档转码为pdf格式,解析开源的pdf数据格式,加工后形成文库自有文档格式,在pc端、无线端排版和渲染。 PC端渲染采用源于PDF的xreader版式数据,版式数据指的是每个元素(文字、图片)都有一个坐标信息和元素的宽高信息,以及其他的描述信息。每一个文本片段、图片和其他矢量元素等根据坐标信息在当前版面固定显示。因此,版式数据比...
相关文章
文章评论
共有0条评论来说两句吧...