当前位置: 首页 > news >正文

乌鲁木齐市做平台网站/太原今日新闻最新头条

乌鲁木齐市做平台网站,太原今日新闻最新头条,网页设计公司的市场定位,用html做一号店网站怎么做原文:Asp.Net Core 轻松学-正确使用分布式缓存前言 本来昨天应该更新的,但是由于各种原因,抱歉,让追这个系列的朋友久等了。上一篇文章 在.Net Core 使用缓存和配置依赖策略 讲的是如何使用本地缓存,那么本篇文章就来了解一下如何…
原文:Asp.Net Core 轻松学-正确使用分布式缓存

前言

    本来昨天应该更新的,但是由于各种原因,抱歉,让追这个系列的朋友久等了。上一篇文章 在.Net Core 使用缓存和配置依赖策略 讲的是如何使用本地缓存,那么本篇文章就来了解一下如何使用分布式缓存,通过本章,你将了解到如何使用分布式缓存,以及最重要的是,如何选择适合自己的分布式缓存;本章主要包含两个部分:

内容提要

  1. 使用 SqlServer 分布式缓存
  2. 使用 Redis 分布式缓存
  3. 实现自定义的分布式缓存客户端注册扩展
  4. 关于本示例的使用说明

1. 使用 SqlServer 分布式缓存

1.1 准备工作,请依照以下步骤实施
  • 1 创建一个 Asp.Net Core MVC 测试项目:Ron.DistributedCacheDemo
  • 2 为了使用 SqlServer 作为分布式缓存的数据库,需要在项目中引用 Microsoft.EntityFrameworkCore 相关组件
  • 3 在 SqlServer 数据库引擎中创建一个数据库,命名为:TestDb
  • 4 打开 Ron.DistributedCacheDemo 项目根目录,执行创建缓存数据表的操作,执行命令后如果输出信息:Table and index were created successfully. 表示缓存表创建成功
dotnet sql-cache create "Server=.\SQLEXPRESS;User=sa;Password=123456;Database=TestDb" dbo AspNetCoreCache

26882-20181222163805468-668209145.png

1.2 开始使用 SqlServer 分布式缓存

.Net Core 中的分布式缓存统一接口是 IDistributedCache 该接口定义了一些对缓存常用的操作,比如我们常见的 Set/Get 方法,而 SqlServer 分布式缓存由 SqlServerCache 类实现,该类位于命名空间 Microsoft.Extensions.Caching.SqlServer 中

  • 在 Startup.cs 中注册分布式缓存
        public void ConfigureServices(IServiceCollection services){services.AddDistributedSqlServerCache(options =>{options.SystemClock = new BLL.LocalSystemClock();options.ConnectionString = this.Configuration["ConnectionString"];options.SchemaName = "dbo";options.TableName = "AspNetCoreCache";options.DefaultSlidingExpiration = TimeSpan.FromMinutes(1);options.ExpiredItemsDeletionInterval = TimeSpan.FromMinutes(5);});...}

上面的方法 ConfigureServices(IServiceCollection services) 中使用 services.AddDistributedSqlServerCache() 这个扩展方法引入了 SqlServer 分布式缓存,并作了一些简单的配置,该配置是由 SqlServerCacheOptions 决定的,SqlServerCacheOptions 的配置非常重要,这里强烈建议大家手动配置

1.3 了解 SqlServerCacheOptions,先来看一下SqlServerCacheOptions 的结构
namespace Microsoft.Extensions.Caching.SqlServer
{public class SqlServerCacheOptions : IOptions<SqlServerCacheOptions>{public SqlServerCacheOptions();// 缓存过期扫描时钟public ISystemClock SystemClock { get; set; }// 缓存过期逐出时间,默认为 30 分钟public TimeSpan? ExpiredItemsDeletionInterval { get; set; }// 缓存数据库连接字符串public string ConnectionString { get; set; }// 缓存表所属架构public string SchemaName { get; set; }// 缓存表名称public string TableName { get; set; }// 缓存默认过期时间,默认为 20 分钟public TimeSpan DefaultSlidingExpiration { get; set; }}
}

该配置非常简单,仅是对缓存使用的基本配置
首先,使用 options.SystemClock 配置了一个本地时钟,接着设置缓存过期时间为 1 分钟,缓存过期后逐出时间为 5 分钟,其它则是连接数据库的各项配置
在缓存过期扫描的时候,使用的时间正是 options.SystemClock 该时钟的时间,默认情况下,该时钟使用 UTC 时间,在我的电脑上,UTC 时间是得到的是美国时间,所以这里实现了一个本地时钟,代码非常简单,只是获取一个本地时间

    public class LocalSystemClock : Microsoft.Extensions.Internal.ISystemClock{public DateTimeOffset UtcNow => DateTime.Now;}
1.4 在控制器中使用分布式缓存
  • 首先使用依赖注入,在 HomeController 中获得 IDistributedCache 的实例对象,该实例对象的实现类型为 SqlServerCache,然后通过 Index 方法增加一项缓存 CurrentTime 并设置其值为当前时间,然后再另一接口 GetValue 中取出该 CurrentTime 的值
    [Route("api/Home")][ApiController]public class HomeController : Controller{private IDistributedCache cache;public HomeController(IDistributedCache cache){this.cache = cache;}[HttpGet("Index")]public async Task<ActionResult<string>> SetTime(){var CurrentTime = DateTime.Now.ToString();await this.cache.SetStringAsync("CurrentTime", CurrentTime);return CurrentTime;}[HttpGet("GetTime")]public async Task<ActionResult<string>> GetTime(){var CurrentTime = await this.cache.GetStringAsync("CurrentTime");return CurrentTime;}}
  • 运行程序,打开地址:http://localhost:5000/api/home/settime,然后查看缓存数据库,缓存项 CurrentTime 已存入数据库中

26882-20181222163817372-1001139527.png

  • 访问接口:http://localhost:5000/api/home/gettime 得到缓存项 CurrentTime 的值

26882-20181222163823764-1218935877.png

  • 等到超时时间过期后,再到数据库查看,发现缓存项 CurrentTime 还在数据库中,这是因为缓存清理机制造成的
1.5 缓存清理

在缓存过期后,每次调用 Get/GetAsync 方法都会 调用 SqlServerCache 的 私有方法 ScanForExpiredItemsIfRequired() 进行一次扫描,然后清除所有过期的缓存条目,扫描方法执行过程也很简单,就是直接执行数据库查询语句

DELETE FROM {0} WHERE @UtcNow > ExpiresAtTime

值得注意的是,在异步方法中使用同步调用不会触发缓存逐出,因为其线程退出导致 Task.Run 未能运行,比如下面的代码

        [HttpGet("GetTime")]public async Task<ActionResult<string>> GetTime(){var CurrentTime = this.cache.GetString("CurrentTime");return CurrentTime;}

将导致 SqlServerCache 无法完整执行方法 ScanForExpiredItemsIfRequired(),因为其内部使用了 Task 进行异步处理,正确的做法是使用 await this.cache.GetStringAsync("CurrentTime");

1.6 关于缓存清理方法 ScanForExpiredItemsIfRequired
        private void ScanForExpiredItemsIfRequired(){var utcNow = _systemClock.UtcNow;if ((utcNow - _lastExpirationScan) > _expiredItemsDeletionInterval){_lastExpirationScan = utcNow;Task.Run(_deleteExpiredCachedItemsDelegate);}}

在多线程环境下,该方法可能除非多次重复扫描,即可能会多次执行 SQL 语句 DELETE FROM {0} WHERE @UtcNow > ExpiresAtTime ,但是,这也仅仅是警告而已,并没有任何可改变其行为的控制途径

1.7 IDistributedCache 的其它扩展方法

.Net Core 中还对 IDistributedCache 进行了扩展,甚至允许通过 Set 方法传入一个 DistributedCacheEntryOptions 以覆盖全局设置,这些扩展方法的使用都比较简单,直接传入相应的值即可,在此不再一一介绍
希望深入研究的同学,可以手动逐一测试

1.8 关于 AddDistributedSqlServerCache() 方法

AddDistributedSqlServerCache 方法内部实际上是进行了一系列的注册操作,其中最重要的是注册了 SqlServerCache 到 IDistributedCache 接口,该操作使得我们可以在控制器中采用依赖注入的方式使用 IDistributedCache 的实例
查看 AddDistributedSqlServerCache 方法的代码片段

 public static IServiceCollection AddDistributedSqlServerCache(this IServiceCollection services, Action<SqlServerCacheOptions> setupAction){if (services == null){throw new ArgumentNullException(nameof(services));}if (setupAction == null){throw new ArgumentNullException(nameof(setupAction));}services.AddOptions();AddSqlServerCacheServices(services);services.Configure(setupAction);return services;}internal static void AddSqlServerCacheServices(IServiceCollection services){services.Add(ServiceDescriptor.Singleton<IDistributedCache, SqlServerCache>());}

2. 使用 Redis 分布式缓存

要在 Asp.Net Core 项目中使用 Redis 分布式缓存,需要引用包:Microsoft.Extensions.Caching.Redis,.Net Core 中的 Redis 分布式缓存客户端由 RedisCache 类提供实现 ,RedisCache 位于程序集 Microsoft.Extensions.Caching.StackExchangeRedis.dll 中,该程序集正是是依赖于大名鼎鼎的 Redis 客户端 StackExchange.Redis.dll,StackExchange.Redis 有许多的问题,其中最为严重的是超时问题,不过这不知本文的讨论范围,如果你希望使用第三方 Redis 客户端替代 StackExchange.Redis 来使用分布式缓存,你需要自己实现 IDistributedCache 接口,好消息是,IDistributedCache 接口并不复杂,定义非常简单

2.1 在 Startup.cs 中注册 Redis 分布式缓存配置
    public void ConfigureServices(IServiceCollection services){services.AddDistributedRedisCache(options =>{options.InstanceName = "TestDb";options.Configuration = this.Configuration["RedisConnectionString"];});...}

注册 Redis 分布式缓存配置和使用 StackExchange.Redis 的方式完全相同,需要注意的是 RedisCacheOptions 包含 3 个属性,而 Configuration 和 ConfigurationOptions 的作用是相同的,一旦设置了 ConfigurationOptions ,就不应该再去设置属性 Configuration 的值,因为,在 AddDistributedRedisCache() 注册内部,会判断如果设置了 ConfigurationOptions 的值,则不再使用 Configuration;但是,我们建议还是通过属性 Configuration 去初始化 Redis 客户端,因为,这是一个连接字符串,而各种配置都可以通过连接字符串进行设置,这和使用 StackExchange.Redis 的方式是完全一致的

2.2 使用缓存
    [Route("api/Home")][ApiController]public class HomeController : Controller{private IDistributedCache cache;public HomeController(IDistributedCache cache){this.cache = cache;}[HttpGet("Index")]public async Task<ActionResult<string>> SetTime(){var CurrentTime = DateTime.Now.ToString();await this.cache.SetStringAsync("CurrentTime", CurrentTime);return CurrentTime;}[HttpGet("GetTime")]public async Task<ActionResult<string>> GetTime(){var CurrentTime = await this.cache.GetStringAsync("CurrentTime");return CurrentTime;}}

细心的你可能已经发现了,上面的这段代码和之前演示的 SqlServerCache 完全一致,是的,仅仅是修改一下注册的方法,我们就能在项目中进行无缝的切换;但是,对于缓存有强依赖的业务,建议还是需要做好缓存迁移,确保项目能够平滑过渡
唯一不同的是,使用 Redis 分布式缓存允许你在异步方法中调用同步获取缓存的方法,这不会导致缓存清理的问题,因为缓存的管理已经完全交给了 Redis 客户端 StackExchange.Redis 了

3. 实现自定义的分布式缓存客户端,下面的代码表示实现一个 CSRedis 客户端的分布式缓存注册扩展

3.1 定义 CSRedisCache 实现 IDistributedCache 接口
    public class CSRedisCache : IDistributedCache, IDisposable{private CSRedis.CSRedisClient client;private CSRedisClientOptions _options;public CSRedisCache(IOptions<CSRedisClientOptions> optionsAccessor){if (optionsAccessor == null){throw new ArgumentNullException(nameof(optionsAccessor));}_options = optionsAccessor.Value;if (_options.NodeRule != null && _options.ConnectionStrings != null)client = new CSRedis.CSRedisClient(_options.NodeRule, _options.ConnectionStrings);else if (_options.ConnectionString != null)client = new CSRedis.CSRedisClient(_options.ConnectionString);elsethrow new ArgumentNullException(nameof(_options.ConnectionString));RedisHelper.Initialization(client);}public void Dispose(){if (client != null)client.Dispose();}public byte[] Get(string key){if (key == null){throw new ArgumentNullException(nameof(key));}return RedisHelper.Get<byte[]>(key);}public async Task<byte[]> GetAsync(string key, CancellationToken token = default(CancellationToken)){if (key == null){throw new ArgumentNullException(nameof(key));}token.ThrowIfCancellationRequested();return await RedisHelper.GetAsync<byte[]>(key);}public void Refresh(string key){throw new NotImplementedException();}public Task RefreshAsync(string key, CancellationToken token = default(CancellationToken)){throw new NotImplementedException();}public void Remove(string key){if (key == null){throw new ArgumentNullException(nameof(key));}RedisHelper.Del(key);}public async Task RemoveAsync(string key, CancellationToken token = default(CancellationToken)){if (key == null){throw new ArgumentNullException(nameof(key));}await RedisHelper.DelAsync(key);}public void Set(string key, byte[] value, DistributedCacheEntryOptions options){if (key == null){throw new ArgumentNullException(nameof(key));}RedisHelper.Set(key, value);}public async Task SetAsync(string key, byte[] value, DistributedCacheEntryOptions options, CancellationToken token = default(CancellationToken)){if (key == null){throw new ArgumentNullException(nameof(key));}await RedisHelper.SetAsync(key, value);}}

代码不多,都是实现 IDistributedCache 接口,然后在 IDisposable.Dispose 中释放资源

3.2 自定义一个配置类 CSRedisClientOptions
    public class CSRedisClientOptions{public string ConnectionString { get; set; }public Func<string, string> NodeRule { get; set; }public string[] ConnectionStrings { get; set; }}

该配置类主要是为 CSRedis 客户端接收配置使用

3.3 注册扩展方法 CSRedisCacheServiceCollectionExtensions
 public static class CSRedisCacheServiceCollectionExtensions{public static IServiceCollection AddCSRedisCache(this IServiceCollection services, Action<CSRedisClientOptions> setupAction){if (services == null){throw new ArgumentNullException(nameof(services));}if (setupAction == null){throw new ArgumentNullException(nameof(setupAction));}services.AddOptions();services.Configure(setupAction);services.Add(ServiceDescriptor.Singleton<IDistributedCache, CSRedisCache>());return services;}}

自定义一个扩展方法,进行配置初始化工作,简化实际注册使用时的处理步骤

3.4 在 Startup.cs 中使用扩展
    public void ConfigureServices(IServiceCollection services){services.AddCSRedisCache(options =>{options.ConnectionString = this.Configuration["RedisConnectionString"];});...}

上面的代码就简单实现了一个第三方分布式缓存客户端的注册和使用

3.5 测试自定义分布式缓存客户端,创建一个测试控制器 CustomerController
    [Route("api/Customer")][ApiController]public class CustomerController : Controller{private IDistributedCache cache;public CustomerController(IDistributedCache cache){this.cache = cache;}[HttpGet("NewId")]public async Task<ActionResult<string>> NewId(){var id = Guid.NewGuid().ToString("N");await this.cache.SetStringAsync("CustomerId", id);return id;}[HttpGet("GetId")]public async Task<ActionResult<string>> GetId(){var id = await this.cache.GetStringAsync("CustomerId");return id;}}

该控制器简单实现两个接口,NewId/GetId,运行程序,输出结果正常

  • 调用 NewId 接口创建一条缓存记录

26882-20181222163837292-2088087825.png

  • 调用 GetId 接口获取缓存记录

26882-20181222163844953-1247530986.png

至此,我们完整的实现了一个自定义分布式缓存客户端注册

4. 关于本示例的使用说明

4.1 首先看一下解决方案结构

26882-20181222163853112-1762202724.png

该解决方案红框处定义了 3 个不同的 Startup.cs 文件,分别是

  1. CSRedisStartup (自定义缓存测试启动文件)
  2. Sql_Startup (SqlServer 测试启动文件)
  3. StackChangeRedis_Startup(StackChange.Redis 测试启动文件)
  • 在使用本示例的时候,通过在 Program.cs 中切换不同的启动文件进行测试
  public static IWebHostBuilder CreateWebHostBuilder(string[] args) =>WebHost.CreateDefaultBuilder(args).UseStartup<Ron.DistributedCacheDemo.Startups.SqlServer.Startup>();

结束语

通过介绍,我们了解到如何在 Asp.Net Core 中使用分布式缓存
了解了使用不同的缓存类型,如 SqlServer 和 Redis
了解到了如何使用不同的缓存类型客户端进行注册
了解到如何实现自定义缓存客户端
还知道了在调用 SqlServer 缓存的时候,异步方法中的同步调用会导致 SqlServerCache 无法进行过期扫描
CSRedisCore 此项目是由我的好朋友 nicye 维护,GitHub 仓库地址:访问CSRedisCore

示例代码下载

https://files.cnblogs.com/files/viter/Ron.DistributedCacheDemo.zip

http://www.jmfq.cn/news/5232637.html

相关文章:

  • wordpress 收藏 插件/百度地图排名怎么优化
  • 劫持别人网站做排名/讯展网站优化推广
  • 网站删除模块/全球中文网站排名
  • discuz 做的网站/关键词查询
  • 用户访问域名劫持网站/淄博搜索引擎优化
  • 朝阳周边网站建设/seo权重优化
  • 高端奶茶加盟网站建设/2021最火营销方案
  • 建设网站时以什么为导向/如何推广自己的产品
  • 鹤山做网站公司/四川seo技术培训
  • 秦皇岛市网站制作公司/百度建站官网
  • 爱做网站/网站收录查询网
  • seo网站推广的主要目的是什么/怎么关键词优化网站
  • 网站开发项目经验总结教训/seo关键词
  • 中国网站建设公司/廊坊网站建设优化
  • 物业公司网站模板/产品宣传推广方式有哪些
  • 网站开发是什么职业/长沙seo外包
  • 网站空间是服务器吗/自媒体平台排名
  • 网站外链建设的八大基本准则/打开百度一下搜索
  • html购物网站源代码/googleplay官网
  • 国内永久免费域名申请网站/小江seo
  • 安阳+网站建设/搜索引擎优化教程
  • 动漫视频网站模板/网络营销公司哪家可靠
  • 百度怎么收录自己的网站/网络竞价托管公司
  • wordpress轻量主题/东莞seo建站投放
  • 一键搭建网站系统/seo是哪个英文的简写
  • 灵山网站建设/优化什么意思
  • 南宁软件优化网站建设/济南网络推广公司电话
  • 怎么查看网站日志/网站域名查询系统
  • 滁州住房与城乡建设官网/潍坊自动seo
  • 钟祥网站开发/个人博客网站模板