gpt4 book ai didi

c# - 向自己注册 IOC 容器

转载 作者:行者123 更新时间:2023-11-30 23:25:24 25 4
gpt4 key购买 nike

假设我有一个接口(interface) IDependencyResolver:

public interface IDependencyResolver{
T Resolve<T>() where T: class;
object Resolve(Type source);
}

以及使用 SimpleInjector 的实现:

public class SIDependencyResolver:IDependencyResolver{
private readonly Container _container;
public SIDependencyResolver(Container container){
_container = container;
_container.Options.DefaultScopedLifestyle = new WcfOperationLifeStyle();
RegisterDependencies(_container, LifeStyle.Scoped);
}
public T Resolve<T>() where T:class{
return _container.GetInstance<T>();
}
public object Resolve(Type source){
return _container.GetInstance(source);
}
public void RegisterDependencies(Container container, LifeStyle lifeStyle){
//register dependencies + register self
}
}

如果我需要在使用服务定位器模式的构造函数中注入(inject) IDependencyResolver,我需要做什么? (是的,我知道……一个反模式……但除此之外)

public class UnitOfWork: IUnitOfWork{
private readonly IDependencyResolver _resolver;
public UnitOfWork(IDependencyResolver resolver){
_resolver = resolver;
}
public IReadRepository<T> Read<T>(){
return _resolver.Resolve<IReadRepository<T>>();
}
public IWriteRepository<T> Write<T>(){
return _resolve.Resolve<IWriteRepository<T>>();
}
}

在过去,我总是将依赖解析器 self 注册为一个单例,所以不使用范围限定的生活方式,因为这给我带来了问题。

container.Register<IDependencyResolver, SIDependencyResolver>(LifeStyle.Singleton);

首先,这是执行此操作的正确方法(例如,在 WCF 范围的情况下,采用单例生活方式)还是有其他方法可以执行此操作?
其次,将 SimpleInjector.Container 传递给我的 dependencyResolver 的构造函数是否正确?

最佳答案

What do I need to do if I need the IDependencyResolver injected in a constructor which uses the Service locator pattern? (yes, I know... an anti-pattern... but this aside)

这不应该被搁置。尽管您像在 UnitOfWork 中那样对容器进行回调的方法很好,(ab)使用这样的IDependencyResolver抽象可以通过代码库快速传播到绝对不合适的地方。

这个IDependencyResolver的用法抽象应限于组合根。但是由于组合根已经知道你正在使用的确切 DI 库的存在,所以有这个 IDependencyResolver抽象与直接依赖于 Container 没有任何好处本身。

所以你的 IUnitOfWork实现应该定义在 inside 你的 Composition Root 中,应该如下所示:

private sealed class SimpleInjectorUnitOfWork : IUnitOfWork {
private readonly Container _container;
public UnitOfWork(Container container){
_container = container;
}
public IReadRepository<T> Read<T>() => _container.GetInstance<IReadRepository<T>>();
public IWriteRepository<T> Write<T>() => _container.GetInstance<IWriteRepository<T>>();
}

注册可以通过以下方式完成:

container.RegisterSingleton<IUnitOfWork>(new SimpleInjectorUnitOfWork(container));

In the past I always registered the dependency resolver self as a singleton, so not with a scoped lifestyle since that gave me problems.

我不清楚你遇到了什么问题,但会造成麻烦的一件事是设置 DefaultScopedLifestyle在构造函数中,而类型由容器自动连接。最好将此调用移出此类构造函数;它使它更清楚:

var container = new Container();
container.Options.DefaultScopedLifestyle = new WcfOperationLifeStyle();

关于c# - 向自己注册 IOC 容器,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37299756/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com