- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我有一个 .Net Core 项目,它注册了一些单例,如下所示:
public void ConfigureServices(IServiceCollection services)
{
services.AddMemoryCache();
services.AddLogging();
services.AddSingleton<IConfiguration>(Configuration);
services.AddSingleton<IDbFactory, DefaultDbFactory>();
services.AddSingleton<IUserRepository, UserRepository>();
services.AddSingleton<IEmailService, EmailService>();
services.AddSingleton<IHostedService, BackgroundService>();
services.AddSingleton<ISettingsRepository, SettingsRepository>();
services.AddSingleton(typeof(TokenManager));
var sp = services.BuildServiceProvider();
var userRepository = sp.GetService<IUserRepository>();
// ...
}
这是唯一注册这些类的地方,没有在任何地方创建其他实例,但是我注意到构造函数被调用了两次。这是为什么?
最佳答案
TL;DR - 如果您需要将注册的单例之一作为配置选项传递给 AddMvc
,不要做我做的事并使用GetService
在ConfigureServices
方法。跳到下面的编辑 2。
我找到了 this answer对于一个类似的问题,但关于服务注册的顺序的答案并不清楚,但它确实指出调用 services.BuildServiceProvider()
构建一个新容器,使服务重新注册。事后看来有点道理......
编辑:
我最初的修复是移动 services.BuildServiceProvider()
在 AddSingleton
之前注册,但事实证明这并不总是有效,正如 Buddha Buddy 指出的那样。
我刚看到 this question它提供了更多关于正在发生的事情的细节。原始注册并没有像我认为的那样被丢弃。我当时想的全错了。
调用 services.BuildServiceProvider()
确实构建了一个新的服务提供者/容器,但这与注册无关。类实现 IUserRepository
当 sp.GetService<IUserRepository>()
时,所有相关服务都由新服务提供者实例化被调用,但是原来的IServiceProvider
保留下来,这就是应用程序的其余部分将使用的内容。
所以当应用程序需要一个 IUserRepository
的实例时(比如因为它是 UsersController
的注入(inject)依赖项,现在已被请求),IUserRepository
服务(及其依赖项,如果有的话)再次被实例化,因为它现在位于原始 IServiceProvider
之下.
上述问题的答案中的评论指出您可以防止这种情况并使用 IServiceProvider
。 , “通过从 ConfigureServices
方法返回服务提供者实例,这样它也将成为您的应用程序使用的容器”。我不确定除了存储指向它的类变量之外你会怎么做,所以它可以在 Configure
中换出方法通过设置 app.ApplicationServices
- 但这对我也不起作用,因为新的服务提供商缺少所有 MVC 服务。
有些人建议使用 app.ApplicationServices
在 Configure()
访问所需的服务,但这对我不起作用,因为我需要在 ConfigureServices
中使用它如下:
//...
serviceProvider = services.BuildServiceProvider();
var userRepository = serviceProvider.GetService<IUserRepository>();
// Add framework services
services.AddMvc(
config =>
{
var policy = new AuthorizationPolicyBuilder()
.RequireAuthenticatedUser()
.Build();
config.Filters.Add(new RolesAuthorizationFilter(userRepository));
});
编辑 2:
找到解决方案! This brilliant post描述了我试图实现的目标并提出了一个非常简洁的解决方案。
如果您需要将对其中一个已注册单例的引用作为 MVC 配置选项传递,而不是尝试在 ConfigureServices
中实例化它,您可以简单地创建一个实现 IConfigureOptions<MvcOptions>
的新类它可以采用注入(inject)的依赖项,将此类注册为单例 - 其他一切都已处理 - 太棒了!
例子:
public void ConfigureServices(IServiceCollection services)
{
services.AddMemoryCache();
services.AddLogging();
services.AddSingleton<IConfiguration>(Configuration);
services.AddSingleton<IDbFactory, DefaultDbFactory>();
services.AddSingleton<IUserRepository, UserRepository>();
services.AddSingleton<IEmailService, EmailService>();
services.AddSingleton<IHostedService, BackgroundService>();
services.AddSingleton<ISettingsRepository, SettingsRepository>();
services.AddSingleton(typeof(TokenManager));
// var sp = services.BuildServiceProvider(); // NO NEED for this after all
// var userRepository = sp.GetService<IUserRepository>();
services.AddMvc(); // No config options required any more
services.AddSingleton<IConfigureOptions<MvcOptions>, ConfigureMvcOptions>(); // Here be the magic...
// ...
}
然后创建新类:
public class ConfigureMvcOptions : IConfigureOptions<MvcOptions>
{
private readonly IUserRepository userRepository;
public ConfigureMvcOptions(IUserRepository userRepository)
{
this.userRepository = userRepository;
}
public void Configure(MvcOptions options)
{
var policy = new AuthorizationPolicyBuilder()
.RequireAuthenticatedUser()
.Build();
options.Filters.Add(new RolesAuthorizationFilter(userRepository));
}
}
多亏了 DI 的魔力,其他一切都得到了处理。我的用户存储库单例及其依赖项仅实例化一次。
关于c# - 如果不实例化两次,就不能在 ConfigureServices 中使用已注册的单子(monad),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51591072/
我有一个问题,想知道如何在已经使用常规登录/注册系统的网站上集成第 3 方登录(也进行静默注册)。 基本上当前登录还是比较规律的: 当用户进入网站 session 时类(class)决定他是否需要重新
我在学习 Hyperledger Fabric 并运行示例代码。 我仍在尝试正确了解事情的运作方式,尤其是在使用证书和加密 Material 的用户/管理员注册和注册中。 我想知道以下如何工作。 1)
我正在尝试使用从我的 Android 设备的 PassWallet 应用程序中保存的票证中获取的 token 向我的设备发送消息。 设备发送使用苹果钱包规范更新通行证所需的所有信息。但是,我正在使用
使用 Passport 本地示例,我可以登录工作。没有关于如何注册用户的文档。 我想为用户提供一个“电子邮件”和“密码”字段,他们可以使用它们来注册该网站。我怎样才能做到这一点?有什么原因没有记录下来
在之前的一些 WSO2IS 版本中,有一个默认的 self 注册功能。但是,我在 5.0 版本中找不到它。 阅读 WSO2IS 5.0 文档,我发现有 2 个用于此功能的 API: getUserId
我已将 Airship SDK 集成到 Android 应用程序中。在应用程序启动和飞艇起飞后,我在日志中得到以下调试信息: 07-27 12:46:31.916 XXX - UALib( 1545)
Delphi 中设计时包的可怕错误之一是以下错误,这意味着注册安装新组件到您的 Palette 上的包: Component TSomething can't be registered by pac
我发现的大部分内容都使用 php 或类似的东西。 我有一个 Angular 前端和 Node/express 服务器代码。还没有后端。我不确定如何继续用户注册。 最佳答案 在没有后端的情况下进行注册等
我正在使用 Drupal 6 开发一个网站。我正在使用我自己的主题,并且一切正常。现在我需要使用我的自定义主题在我的网站上创建一个自定义登录/注册表单。我尝试了很多方法,但一切都重定向到我的管理主题,
我在运行此代码时收到 Sip 异常。这是因为 manager.register(me,20,listener)。 下面是我的代码,所以请帮我更正这段代码。我正在使用 SipDemo 代码注册我的帐户。
在过去的几天里,我一直在尝试为基于 VUE 的 excel 制作任务 Pane 插件。 我已按照 link 的指南进行操作我试图为 onSelectionChange 注册一个事件处理程序。它已经有些
我需要在使用 django-registration 应用程序的登录表单中实现一个“记住我”按钮。任何ane可以帮助我向我展示这样做的方法吗? 谢谢 最佳答案 一种方法是更改 session 到期
我发现,如果您使用 Django 1.5 版本,则 django 注册模块会中断,因为在最新的 django 开发版本中,simple.py 类已被删除。 最佳答案 此问题现已修复: hg clone
我正在尝试将 facebook connect 实现到我的网站,但有几个问题。 1:是否可以使用用户当前的 Facebook 电子邮件/密码在我的网站上注册用户。 假设用户点击链接通过 faceboo
我使用 Facebook 注册来允许人们在我的网站上注册。有没有可能,在注册后,他在我的网站注册的成员(member)墙上会张贴? 最佳答案 这可能不是您正在寻找的答案,但我强烈建议您不要这样做。用户
I would like to use a slash (/) for a search during a vimscript, but I don't want to overwrite the "
我正在使用 jqgrid 并且有显示日期的列,但是来自服务器的日期以 json 格式出现,如下所示, "CommentedDate": "\/Date(1304324941000+0530)\/" 如
我希望用户可以直接登录主页,而不是在“../account/login/”页面上登录。我应该做什么才能使它成为可能?如何将主页上的输入字段与 allauth 连接?我不知道这样是否太复杂而无法以这种方
This question already has answers here: Understanding NSString comparison (7个答案) 5年前关闭。 我正在尝试制作注册表。有
我正在使用 Django 注册。它提供了处理 registration_form.html 的 View ,该 html 当前包含用户名、密码 1、密码 2 和电子邮件作为我的应用程序中的用户可输入字
我是一名优秀的程序员,十分优秀!