asp.net-core - netcore 2.1 具有范围存储库的分布式缓存
问题描述
需要一些帮助。
我有一个 .netcore 2.1 API,它通过来自其客户端的 Azure Bearer 令牌进行保护。我想从客户端的不记名令牌中收集用户信息并将其存储在 SQL 数据库中,以便在添加/删除/编辑等时标记数据库中的条目。因此,对于 SQL 表连接,我需要用户SQL 中的信息。
下面是我使用 IDistributedCache 实现的缓存服务。在初始化时,我试图将所有当前存储的用户从 SQL DB 加载到缓存中,然后在新用户连接时添加到缓存中。
为了捕获整个 API 之间的连接,我使用 TypeFilterAttribute 来执行 OnActionExecuting。
问题是 CacheService 是一个单例并且正在调用 UserRepository - 它是作用域的。这是不允许的。
有什么想法吗?
启动.cs
public void ConfigureServices(IServiceCollection services)
{
...
// Context
services.TryAddSingleton<IHttpContextAccessor, HttpContextAccessor>();
services.TryAddSingleton<CacheService>();
// Repositories
services.TryAddScoped<IUserRepository, UserRepository>();
services.AddDistributedMemoryCache();
services.AddMvc(
opts => opts.Filters.Add(new HttpInterceptor())
)
...
缓存服务.cs
public class CacheService
{
private readonly IDistributedCache _cache;
private readonly IUserRepository _userRepository;
public CacheService(
IDistributedCache cache,
[FromServices] IUserRepository userRepository
)
{
_cache = cache;
_userRepository = userRepository;
// Populate cache from DB
var users = _userRepository.GetAll().Result;
foreach (var u in users)
{
if (_cache.GetAsync(u.Username).Result == null)
{
var profileSerialised = JsonConvert.SerializeObject(UserToUserProfile(u));
var entry = Encoding.UTF8.GetBytes(profileSerialised);
_cache.SetAsync(u.Username, entry, new DistributedCacheEntryOptions { AbsoluteExpirationRelativeToNow = TimeSpan.FromMinutes(30) });
}
}
}
HttpInterceptor.cs
public class HttpInterceptor : TypeFilterAttribute
{
public HttpInterceptor() : base(typeof(IHttpInterceptor))
{
}
private class IHttpInterceptor : IActionFilter
{
private readonly CacheService _cache;
private readonly IUserRepository _userRepository;
public IHttpInterceptor(
CacheService cache,
IUserRepository userRepository)
{
_cache = cache;
_userRepository = userRepository;
}
public void OnActionExecuting(ActionExecutingContext context)
{
if (context.HttpContext.User.Identity.IsAuthenticated)
{
this._cache.GetUserProfile(context.HttpContext.User.Identity.Name);
}
}
解决方案
首先,你正在看这个颠倒和倒退。让一些服务将内容添加到缓存中,然后让其他代码只是假设缓存中的内容已准备就绪,这是灾难的根源。相反,让您的依赖代码逐字地请求它需要的数据,然后,如果您想缓存它,请执行获取数据的方法。这样,您的应用程序代码就不会知道数据的来源;它只是调用一个方法并获取它想要的数据。在幕后,它要么从数据库中提取,要么从缓存中提取,具体取决于哪个可用/首选。
无论如何,您的缓存服务存在严重问题。首先,它首先不应该是单例。没有理由这样做,而且由于您正在处理内部的范围服务,因此您只会使事情变得比他们需要的更困难。其次,你永远不应该在构造函数中使用 I/O。只有简单的变量/道具初始化应该在那里完成。任何需要实际工作的东西都应该进入方法。如果你真的想在初始化时做点什么,那么你应该实现一个工厂模式。例如,您可能有类似 aCacheServiceFactory
的Create
方法,它返回完全实例化的方法,CacheService
包括调用任何执行实际工作的方法。
撇开免责声明不谈,一般来说,要在单例中使用作用域服务,您必须创建一个作用域。每次您想使用该服务时都必须这样做;您不能将服务持久化到单例类上的 ivar。简单地说,你注入IServiceProvider
你的类,它本身是单例范围的,所以你不会有任何问题。然后,当您需要使用范围服务时:
using (var scope = provider.CreateScope())
{
var repo = scope.ServiceProvider.GetRequiredService<IUserRepository>();
// do something with repo
}
这称为服务定位器反模式。之所以这样称呼它,是因为您确实应该避免这样做。有时这并不总是可能的。但是,通常情况下,您可以简单地以不同的方式设计事物:例如使服务本身限定范围。
推荐阅读
- c# - 如何从 ssis 的文件夹中排除特定文件?
- excel - 用户窗体文本框输入
- robotframework - 如果在机器人框架上如何使用 run 关键字
- pyspark - 如何在pyspark中将日期格式'YYYY-MM-DD'转换为ddMMyy?
- reactjs - React:我应该使用什么来代替“key”道具中的数组索引?
- debugging - LLDB 断点性能 - 我应该期待什么?
- kubernetes-helm - Helm: Get value from a Map where the key is variable
- swiftui - SwiftUI ForEach 获取二维数组中的元素和索引
- performance - Rasa NLU 模型加载需要大量时间
- python - 从多个部分数据框创建熊猫数据框