Зарегистрировать компонент на основе имени параметра в запросчике в windsor - PullRequest
4 голосов
/ 18 февраля 2012

У меня есть этот интерфейс для использования AutoMapper:

public interface IMapper
{
    object Map(object source, Type sourceType, Type destinationType);
}

Тогда для каждого типа данных у меня есть свой класс сопоставления, например:

 public class UserMapper : IMapper
{
    static UserMapper()
    {
        Mapper.CreateMap<User, UserViewModel>();
        Mapper.CreateMap<UserViewModel, User>();
    }

    public object Map(object source, Type sourceType, Type destinationType)
    {
        return Mapper.Map(source, sourceType, destinationType);
    }
}

Тогда у меня есть IMapper как один из параметров в моем классе контроллера, как это:

public UsersController(IUsersRepository repo, IMapper userMapper)
{....}

Я использую Windsor в качестве IOC для своего приложения, и проблема в том, что я хочу зарегистрировать компоненты, чтобы при запуске в UsersController он использовал класс UserMapper, а при работе с ProductsController он использовал мой класс ProductMapper.

Мой регистрационный код выглядит примерно так:

container.Register(
    Component.For<IMapper>()
             .ImplementedBy<UsersMapper>()
             .Named("usersMapper"),
    Component.For<IMapper>()
             .ImplementedBy<ProductsMapper>()
             .Named("productsMapper"),
    Component.For<ProductController>()
             .ServiceOverrides(ServiceOverride.ForKey("usersMapper").Eq("productsMapper"))
)

Я сделал свою домашнюю работу в google и stackoverflow, и я знаю, что мне нужно использовать ServicesOverride, но я все еще застрял в этом, может кто-нибудь помочь мне, пожалуйста?

Спасибо

Ответы [ 2 ]

8 голосов
/ 19 февраля 2012

Хотя решение svick мне кажется правильным (хотя я не пытался его скомпилировать), этот сценарий является отличным примером для основанной на соглашении конфигурации .

Давайте введем это соглашение: Каждый потребитель IMapper сообщит предполагаемую роль картографа своим именем. По умолчанию это имя будет соответствовать типу с тем же именем - только с другим регистром.

Итак, параметры конструктора могут отображаться так:

  • userMapper -> UserMapper
  • productMapper -> ProductMapper

В Castle Windsor такая конфигурация может выглядеть следующим образом:

container.Register(Classes
    .FromThisAssembly()
    .Pick()
    .WithServiceAllInterfaces()
    .WithServiceSelf());

container.Kernel.Resolver.AddSubResolver(
    new MapperConvention(container.Kernel));

И Sub Resolver (где волшебство действительно происходит) выглядит так:

public class MapperConvention : ISubDependencyResolver
{
    private readonly IKernel kernel;

    public MapperConvention(IKernel kernel)
    {
        this.kernel = kernel;
    }

    public bool CanResolve(CreationContext context,
        ISubDependencyResolver contextHandlerResolver,
        ComponentModel model,
        DependencyModel dependency)
    {
        return typeof(IMapper).IsAssignableFrom(dependency.TargetType);
    }

    public object Resolve(CreationContext context,
        ISubDependencyResolver contextHandlerResolver,
        ComponentModel model,
        DependencyModel dependency)
    {
        var representativeMapperType = typeof(UserMapper);
        var concreteMapperType = representativeMapperType.Assembly
            .GetExportedTypes()
            .Where(t => 
                t.Name.Equals(dependency.DependencyKey,
                    StringComparison.OrdinalIgnoreCase))
            .Single();
        return this.kernel.Resolve(concreteMapperType);
    }
}
7 голосов
/ 18 февраля 2012

У меня эта регистрация работает:

container.Register(
    Component.For<IMapper>()
        .ImplementedBy<UserMapper>()
        .Named("userMapper"),
    Component.For<IMapper>()
        .ImplementedBy<ProductMapper>()
        .Named("productMapper"),
    Component.For<UsersController>()
        .ServiceOverrides(ServiceOverride.ForKey<IMapper>().Eq("userMapper")),
    Component.For<ProductsController>()
        .ServiceOverrides(ServiceOverride.ForKey<IMapper>().Eq("productMapper"))
    );
...