c# - Autofac:注册 Func<> 或工厂?

标签 c# inversion-of-control ioc-container autofac

我必须根据从服务器接收到的一些消息/属性在运行时创建实现,这些消息/属性也需要由新创建的对象进行转换。我是 Autofac 的新手,但据我所知,有两种方法可以解决这个问题。

方法一:注册专用工厂

...
builder.RegisterType<MTextField>().Keyed<IComponent>(typeof(TextFieldProperties));
builder.RegisterType<ComponentFactory>().As<IComponentFactory>();

public class ComponentFactory : IComponentFactory
{
    private readonly IIndex<Type, IComponent> _lookup;

    public ComponentFactory(IIndex<Type, IComponent> lookup)
    {
        _lookup = lookup;
    }

    public IComponent Create(ComponentProperties properties)
    {
        var component = _lookup[properties.GetType()];
        component.Transform(properties);
        return component;
    }
}

方法二:根据函数注册

...
builder.RegisterType<MTextField>().Keyed<IComponent>(typeof(TextFieldProperties));
builder.Register<Func<ComponentProperties, IComponent>>(c =>
{
    var context = c.Resolve<IComponentContext>();
    return properties =>
    {
        var component = context.ResolveKeyed<IComponent>(properties.GetType());
        component.Transform(properties);
        return component;
    };
});

问题:

我认为这可能是一件主观的事情,但我还是想问一下。

  • 哪种方法更可取,为什么?
  • 还有更好的解决方案吗?
  • 是否真的有必要在“方法 2”中存储上下文?

编辑

好的,我用 autofac 玩了更多。这是我目前的方法:

public class TransformerFactory<D, T> : ITransformFactory<D, T>
    where T : ITransform<D>
{
    private readonly IIndex<Type, T> _lookup;


    public TransformerFactory(IIndex<Type, T> lookup)
    {
        _lookup = lookup;
    }


    public T Create(D data, Action<T> prepareInstance = null)
    {
        var instance = _lookup[data.GetType()];
        if (prepareInstance != null)
        {
            prepareInstance(instance);
        }
        instance.Transform(data);
        return instance;
    }
}

builder.RegisterGeneric(typeof(TransformerFactory<,>)).As(typeof(ITransformFactory<,>)); 
// e.g. var x = container.Resolve<ITransformFactory<ComponentProperties, IComponent>>();

最佳答案

第一种方法似乎是更可取的方法。我提出的两个原因是:

  1. Keyed 和 IIndex 为手头的任务提供了充分且干净的开箱即用解决方案。而方法 2 使用的是一种更通用的工具,它需要您执行额外的逻辑才能工作(即您编写代码来调用 ResolveKeyed)。如果有一个简单的特定用途解决方案,请优先使用它而不是更通用的解决方案。

  2. 方法 1 将能够正确管理生命周期和范围。正如您提出的那样,方法 2 捕获上下文,并将根据该上下文解决各个实例。这意味着各个实例的生命周期将基于工厂的范围和生命周期,而不是为各个服务指定的生命周期策略。

关于c# - Autofac:注册 Func<> 或工厂?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/9546387/

相关文章:

mvvm - Autofac MVVM-终身

c# - AutoMapper:IDataReader 和 DTO 对象之间的映射

c# - 返回 List<object> 时 Asp.net Webservice 不工作

java - 依赖注入(inject)只针对服务类型对象和单例吗? (而不是图形用户界面?)

c# - 在 C# 控制台应用程序中正确使用 Autofac

c# - 统一解决同一类型的多个实例

c# - 如何为 RadioButton 组注册事件?

c# - Windows 8 商店应用程序不支持 System.Threading.Thread

oop - 如何处理难以表达的依赖要求?

java - 如何使用 PicoContainer 管理动态依赖项?