加入收藏 | 设为首页 | 会员中心 | 我要投稿 李大同 (https://www.lidatong.com.cn/)- 科技、建站、经验、云计算、5G、大数据,站长网!
当前位置: 首页 > 百科 > 正文

c# – 使用Ninject(或其他容器)如何找出请求服务的类型?

发布时间:2020-12-16 01:30:17 所属栏目:百科 来源:网络整理
导读:假设我有一个服务接口: public interface IFooService{ void DoSomething();} 并且该服务的具体实现是通用的: public class FooServiceTRequestingClass : IFooService{ public virtual void DoSomething() { }} 我还有一些需要IFooService实例的其他类:
假设我有一个服务接口:

public interface IFooService
{
   void DoSomething();
}

并且该服务的具体实现是通用的:

public class FooService<TRequestingClass> : IFooService
{
   public virtual void DoSomething() { }
}

我还有一些需要IFooService实例的其他类:

public class Bar
{
   private IFooService _fooService;
   public Bar(IFooService fooService)
   {
      this._fooService = fooService;
   }
}

我需要连接我的IoC容器,这样当创建Bar时,它会传递一个FooService< Bar>的构造函数参数.还有很多其他课程就像Bar一样.每个人都可能需要一个FooService实例< TRequestingClass>传递给他们,其中TRequestingClass是需要IFooService实例的类的类型.我不需要向IFooService的消费者公开这个怪癖.他们应该关心的是他们可以调用他们传递的IFooService的方法.他们不应该知道他们传递的IFooService的具体实现需要任何特殊的构造.

FooService< T>的可接受的替代方案.将是一个非泛型类,在其构造函数中包含一个字符串参数,该类包含要为其创建的类的名称.即:

public class FooService : IFooService
{
   public FooService(string requestingClassName) { }
}

如何通过这种方式连接我的IoC容器来构建依赖项?

如果你为什么我想要这样一个奇怪的结构而感到困惑,那么考虑当你得到一个用log4net.LogManager.GetLogger(typeof(SomeClass))创建的ILog时log4net如何工作得最好.我不想通过引用log4net来丢弃我的代码,所以我想编写一个简单的ILogger接口,并用这样的方式实现它:

public class GenericLogger<T> : ILogger
{
    private readonly ILog log;

    public GenericLogger()
    {
        this.log = log4net.LogManager.GetLogger(typeof(T));
    }

    public void Debug(object message)
    {
        this.log.Debug(message);
    }

    /* .... etc ....  */
}

解决方法

最简单的方法是创建ILogger< T>.接口:

public class ILogger<T> : ILogger { }
public class GenericLogger<T> : ILogger<T> { ... }

然后依靠泛型类型推断来获得正确的类型.例如,在Ninject中,您需要以下绑定:

Bind(typeof(ILogger<>)).To(typeof(GenericLogger<>));

那么您的消费类型将如下所示:

public class FooService : IFooService {
  public FooService(ILogger<FooService> logger) { ... }
}

如果你坚决反对ILogger< T>界面,您可以做一些更有创意的事情,比如自定义提供程序,它会读取IContext以确定父类型.

public class GenericLogger : ILogger {
  public class GenericLogger(Type type) { ... }
}

public class LoggerProvider : Provider<ILogger> {
  public override ILogger CreateInstance(IContext context) {
    return new GenericLogger(context.Target.Member.ReflectedType);
  }
}

然后消费类型将如下工作:

public class FooService : IFooService {
  public FooService(ILogger logger) { ... }
}

(编辑:李大同)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    推荐文章
      热点阅读