Autofac命名注册构造函数注入

时间:2014-06-02 01:59:11

标签: dependency-injection inversion-of-control autofac

Autofac是否支持在组件的构造函数中指定注册名称?

示例:Ninject的NamedAttribute

2 个答案:

答案 0 :(得分:19)

您需要在顶部使用Autofac.Extras.Attributed包来实现此目的

所以,让我们说你有一个界面和两个班级:

public interface IHello
{
    string SayHello();
}

public class EnglishHello : IHello
{
    public string SayHello()
    {
        return "Hello";
    }
}

public class FrenchHello : IHello
{
    public string SayHello()
    {
        return "Bonjour";
    }
}

然后你有一个消费者类,你想要选择注入哪个实例:

public class HelloConsumer
{
    private readonly IHello helloService;

    public HelloConsumer([WithKey("EN")] IHello helloService)
    {
        if (helloService == null)
        {
            throw new ArgumentNullException("helloService");
        }
        this.helloService = helloService;
    }

    public string SayHello()
    {
        return this.helloService.SayHello();
    }
}

注册和解决:

ContainerBuilder cb = new ContainerBuilder();

cb.RegisterType<EnglishHello>().Keyed<IHello>("EN");
cb.RegisterType<FrenchHello>().Keyed<IHello>("FR");
cb.RegisterType<HelloConsumer>().WithAttributeFilter();
var container = cb.Build();

var consumer = container.Resolve<HelloConsumer>();
Console.WriteLine(consumer.SayHello());

注册这样的消费者时不要忘记AttributeFilter,否则解析将失败。

另一种方法是使用lambda而不是属性。

cb.Register<HelloConsumer>(ctx => new HelloConsumer(ctx.ResolveKeyed<IHello>("EN")));

我发现第二个选项更干净,因为你也避免在你的项目中引用autofac程序集(只是为了导入一个属性),但那部分当然是个人意见。

答案 1 :(得分:0)

        var builder = new ContainerBuilder();

        builder.RegisterType<LiveGoogleAnalyticsClass>().Named<IGoogleAnalyticsClass>("Live");
        builder.RegisterType<StagingGoogleAnalyticsClass>().Named<IGoogleAnalyticsClass>("Staging");
        var container = builder.Build();

        var liveGAC = container.ResolveNamed<IGoogleAnalyticsClass>("Live");
        var stagingGAC = container.ResolveNamed<IGoogleAnalyticsClass>("Staging");

这只需要标准的Autofac dll,不需要Autofac.Extras.Attributed组件的引用。由于遗留的要求,我目前正在使用Autofac 3.5.0。