【问题标题】:Resolve dependency based on configuration根据配置解决依赖
【发布时间】:2015-05-27 09:08:01
【问题描述】:

在我的 Windows 服务应用程序中,我需要在服务启动时使用配置解析组件。我使用 Castle Windsor 作为我的 IoC 容器。

应用程序看起来像:

public class RootComponent : IRootComponent {
    public RootComponent (IDataProvider1 provider1, IDataProvider2 provider2)
    {
        this.provider1 = provider1;
        this.provider2 = provider2;
    }

    ...
}

public class DataProvider1 : IDataProvider1
{
    ...

    public DataProvider1 (IDbHelper dbHelper)
    {
        this.dbHelper = dbHelper;
    }

    ...
}

public class DataProvider2 : IDataProvider2
{
    ...

    public DataProvider1 (IDbHelper dbHelper)
    {
        this.dbHelper = dbHelper;
    }

    ...
}

public interface IDbHelper
{
    IDbConnection GetNewConnection();
    DbParameter CreateDbParameter(string paramName, object paramValue);
    string GetCommandString(string commandName);
}

public class MsSqlDbHelper : IDbHelper
{
    ...

    public MsSqlDbHelper(string connectionString)
    {
        this.connectionString = connectionString;
    }

    ...
}

public class PostgreDbHelper : IDbHelper
{
    ...

    public PostgreDbHelper(string connectionString)
    {
        this.connectionString = connectionString;
    }

    ...
}

我想在服务构造函数中注册依赖项,并在 OnStart 方法中读取配置并根据它解析正确的 IDbHelper:

public partial class MyWindowsService : ServiceBase
{
    public MyWindowsService()
    {
        InitializeComponent();

        this.container = new WindsorContainer();
        RegisterDependencies();
    }

    private void RegisterDependencies()
    {
        container.Register(
            Classes.FromAssemblyInThisApplication().BasedOn<IRootComponent>().WithServiceFromInterface(), 
            Classes.FromAssemblyInThisApplication().BasedOn<IDataProvider1>().WithServiceFromInterface(),
            Classes.FromAssemblyInThisApplication().BasedOn<IDataProvider2>().WithServiceFromInterface(),
            Classes.FromAssemblyInThisApplication().BasedOn<IDbHelper>().WithServiceFromInterface()
                .ConfigureFor<MsSqlDbHelper>(
                    registration => {
                        registration.DependsOn(Dependency.OnValue<string>(ConnectStringProvider.GetConnectionString("connectString1")));
                        registration.Named("msSql");
                    })
                .ConfigureFor<PostgreDbHelper>(
                    registration => {
                        registration.DependsOn(Dependency.OnValue<string>(ConnectStringProvider.GetConnectionString("connectString2")));
                        registration.Named("postgreSql");
                    }));
    }

    protected override void OnStart(string[] args)
    {
        ResolveDependencies();
    }

    private void ResolveDependencies()
    {
        // Helper properties contain "msSql" or "postgreSql" value
        root = container.Resolve<IRootComponent>(Config.Provider1Helper, Config.Provider2Helper);
    }

    ...
}

我看到三个选项如何解决配置的 IDbHepler:

  1. 类型化工厂设施
  2. 儿童容器
  3. 实现 IHandlerSelector

什么是最好的方法,为什么?

【问题讨论】:

    标签: dependency-injection castle-windsor ioc-container


    【解决方案1】:

    解决方案比我预期的要简单。 DynamicParameters 是我需要的:

    private void RegisterDependencies()
    {
        container.Register(
            Classes.FromAssemblyInThisApplication().BasedOn<IRootComponent>().WithServiceFromInterface(), 
            Classes.FromAssemblyInThisApplication().BasedOn<IDataProvider1>().WithServiceFromInterface()
                .Configure(registration => registration.DynamicParameters((kernel, parameters) =>
                        {
                            IDbHelper dbHelper = kernel.Resolve<IDbHelper>(Config.Provider1Helper);
                            parameters.Add("dbHelper", dbHelper);
                            return k => k.ReleaseComponent(dbHelper);
                        })),
            Classes.FromAssemblyInThisApplication().BasedOn<IDataProvider2>().WithServiceFromInterface(),
                .Configure(registration => registration.DynamicParameters((kernel, parameters) =>
                        {
                            IDbHelper dbHelper = kernel.Resolve<IDbHelper>(Config.Provider2Helper);
                            parameters.Add("dbHelper", dbHelper);
                            return k => k.ReleaseComponent(dbHelper);
                        })),
            Classes.FromAssemblyInThisApplication().BasedOn<IDbHelper>().WithServiceFromInterface()
                .ConfigureFor<MsSqlDbHelper>(
                    registration => {
                        registration.DependsOn(Dependency.OnValue<string>(ConnectStringProvider.GetConnectionString("connectString1")));
                        registration.Named("msSql");
                    })
                .ConfigureFor<PostgreDbHelper>(
                    registration => {
                        registration.DependsOn(Dependency.OnValue<string>(ConnectStringProvider.GetConnectionString("connectString2")));
                        registration.Named("postgreSql");
                    }));
    }
    
    protected override void OnStart(string[] args)
    {
        // Helper properties should contain "msSql" or "postgreSql" value
        // take configuration from arguments
        Config.Provider1Helper = UseMsSql(args, 0) ? "msSql" : "postgreSql";
        Config.Provider2Helper = UseMsSql(args, 1) ? "msSql" : "postgreSql";
    
        ResolveDependencies();
    }
    
    private void ResolveDependencies()
    {
        root = container.Resolve<IRootComponent>();
    }
    
    private bool UseMsSql(string[] args, int argNumber)
    {
        ...
    }
    

    【讨论】:

      猜你喜欢
      • 2022-12-20
      • 1970-01-01
      • 1970-01-01
      • 2015-06-04
      • 1970-01-01
      • 1970-01-01
      • 2013-02-14
      • 2011-04-01
      • 2023-03-13
      相关资源
      最近更新 更多