关于我 作者博客|文章首发 缓存基础知识 缓存可以减少生成内容所需的工作,从而显著提高应用程序的性能和可伸缩性。 缓存最适用于不经常更改的 数据,生成 成本很高。 通过缓存,可以比从数据…
                                                                                                                                                                                    ### 关于我 
作者博客|文章首发
缓存基础知识
缓存可以减少生成内容所需的工作,从而显著提高应用程序的性能和可伸缩性。 缓存最适用于不经常更改的 数据,生成 成本很高。 通过缓存,可以比从数据源返回的数据的副本速度快得多。 应该对应用进行编写和测试,使其 永不 依赖于缓存的数据。
ASP.NET Core 支持多个不同的缓存。 最简单的缓存基于 IMemoryCache。
表示存储在 web 服务器的内存中的缓存。 在服务器场上运行的应用 (多台服务器) 应确保会话在使用内存中缓存时处于粘滞状态。 粘滞会话确保来自客户端的后续请求都将发送到相同的服务器。
内存中缓存可以存储任何对象。 分布式缓存接口仅限
。 内存中和分布式缓存将缓存项作为键值对。
缓存指南
代码应始终具有回退选项,以获取数据,而 不是依赖于可用的缓存值。
缓存使用稀有资源内存,限制缓存增长:
不要 使用外部 输入作为缓存键。
使用过期限制缓存增长。
使用 SetSize、Size 和 SizeLimit 限制缓存大小]。 ASP.NET Core 运行时不会根据内存 压力限制缓存 大小。 开发人员需要限制缓存大小。
使用
DI注入
创建一个NetCore控制台项目,进行缓存的项目演示。
控制台项目只有一个初始化的Program.cs文件。基于NetCore进行项目编码,每一步就是创建一个基础模板,使用依赖注入的方式。
| 12
 
 | nuget install Microsoft.Extensions.Hosting
 
 | 
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 
 | public static class Program{
 static async void Main(string[] args)
 {
 var builder = new HostBuilder().ConfigureServices((context, service) =>
 {
 
 });
 
 await builder.RunConsoleAsync();
 }
 }
 
 
 | 
注入缓存服务,控制台需要下载库 Microsoft.Extensions.Caching.Memory
| 12
 
 | nuget install Microsoft.Extensions.Caching.Memory
 
 | 
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 
 | 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();
 }
 }
 
 
 | 
后台服务
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 
 | 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
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 
 |   public class CacheService{
 private readonly IMemoryCache _memoryCache;
 
 public CacheService(IMemoryCache memoryCache)
 {
 _memoryCache = memoryCache;
 }
 }
 
 
 | 
最基本的使用
Set方法根据Key设置缓存,默认缓存不过期
Get方法根据Key取出缓存
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 
 |   
 
 public void BaseCache()
 {
 string cacheKey = "timestamp";
 
 _memoryCache.Set(cacheKey, DateTime.Now.ToString());
 
 
 Console.WriteLine(_memoryCache.Get(cacheKey));
 }
 
 
 | 
IMemoryCache提供一些好的语法糖供开发者使用,具体内容看下方文档
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 
 |   
 
 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秒后过期
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 
 |   string cacheKey = "absoluteKey";
 _memoryCache.Set(cacheKey, DateTime.Now.ToString(), TimeSpan.FromSeconds(5));
 
 
 for (int i = 0; i < 6; i++)
 {
 Console.WriteLine(_memoryCache.Get(cacheKey));
 Thread.Sleep(1000);
 }
 
 
 | 
滑动到期
过期策略 2秒的滑动过期时间,如果2秒内有访问,过期时间延后。当2秒的区间内没有访问,缓存过期
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 
 |   string cacheSlibingKey = "slibingKey";
 MemoryCacheEntryOptions options = new MemoryCacheEntryOptions();
 options.SlidingExpiration = TimeSpan.FromSeconds(2);
 
 _memoryCache.Set(cacheSlibingKey, DateTime.Now.ToString(), options);
 
 
 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秒滑动过期
满足任意一个缓存都将失效
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 
 |   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);
 
 
 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----------------");
 
 
 | 
缓存状态变化事件
当缓存更新、删除时触发一个回调事件,记录缓存变化的内容。
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 
 |   
 
 public void CacheStateCallback()
 {
 MemoryCacheEntryOptions options = new MemoryCacheEntryOptions();
 options.AbsoluteExpiration = DateTime.Now.AddSeconds(3
 );
 options.RegisterPostEvictionCallback(MyCallback, this);
 
 
 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);
 }
 
 
 | 
缓存依赖策略
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 
 |   
 
 public void CacheDependencyPolicy()
 {
 string DependentCTS = "DependentCTS";
 string cacheKeyParent = "CacheKeys.Parent";
 string cacheKeyChild = "CacheKeys.Child";
 
 var cts = new CancellationTokenSource();
 _memoryCache.Set(DependentCTS, cts);
 
 
 using (var entry = _memoryCache.CreateEntry(cacheKeyParent))
 {
 
 entry.Value = "parent" + DateTime.Now;
 
 
 entry.RegisterPostEvictionCallback(MyCallback, this);
 
 
 _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);
 
 
 _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);
 }
 
 
 | 
参考资料
AspNetCore中的缓存内存
.NetCore缓存篇之MemoryCache
Asp.Net Core 轻松学-在.Net Core 使用缓存和配置依赖策略
拥抱.NET Core系列:MemoryCache 缓存过期
推荐阅读
Redis��具收费后新的开源已出现
GitHub上Star最高的工程师技能图谱
中国程序员最容易发错的单词
推荐!!! Markdown图标索引网站
最后
本文到此结束,希望对你有帮助 😃
如果还有什么疑问或者建议,可以多多交流,原创文章,文笔有限,才疏学浅,文中若有不正之处,万望告知。
更多精彩技术文章汇总在我的 公众号【程序员工具集】,持续更新,欢迎关注订阅收藏。

本文标题: NetCore的缓存使用详例
本文作者: OSChina
发布时间: 2021年04月15日 09:46
最后更新: 2025年07月13日 05:44
原始链接: https://haoxiang.eu.org/b351a9d2/
版权声明: 本文著作权归作者所有,均采用CC BY-NC-SA 4.0许可协议,转载请注明出处!