0

我的 Ninject 模块中有以下代码重复了几次。我可以使用哪些方法和技术来减少这种重复代码?

public override void Load()
{
    Bind<IDataReader<IList<Price>>>()
        .To<PricesDataReader>().Named("ValDatePrices");
    Bind<IDataConnection<IList<PricesCsvRecord>>>()
        .To<PricesXLConnection>().WhenParentNamed("ValDatePrices")
        .Named("ValDatePricesXLConnection");
    Bind<IDirectoryBuilder>()
        .ToMethod(DefaultValDatePricesDirectory)
        .WhenParentNamed("ValDatePricesXLConnection");

    Bind<IDataReader<IList<Price>>>()
        .To<PricesDataReader>().Named("EDDatePrices");
    Bind<IDataConnection<IList<PricesCsvRecord>>>()
        .To<PricesXLConnection>().WhenParentNamed("EDDatePrices")
        .Named("EDDatePricesXLConnection");
    Bind<IDirectoryBuilder>()
        .ToMethod(DefaultEDDatePricesDirectory)
        .WhenParentNamed("EDDatePricesXLConnection");
}

主要区别发生在请求IDirectoryBuilder其主要功能是通过使用实现根据配置设置确定文件位置时IDirectory

在上面的示例中,我返回 a DefaultDirectoryBuilder,但是我在下面有几个其他的实现(参见EdNrrDirectoryBuilder方法)。

public IDirectoryBuilder DefaultValDatePricesDirectory(IContext arg) 
{
    return new DefaultDirectoryBuilder(
         ConfigurationManager.AppSettings["VALDATE_PRICES_DIR"],
         ConfigurationManager.AppSettings["VALDATE_PRICES_FILENAME"]);
}

public IDirectoryBuilder DefaultEDDatePricesDirectory(IContext arg) 
{
    return new DefaultDirectoryBuilder(
         ConfigurationManager.AppSettings["EDDATE_PRICES_DIR"],
         ConfigurationManager.AppSettings["EDDATE_PRICES_FILENAME"]);
}

public IDirectoryBuilder EdNrrDirectoryBuilder(IContext arg)
{
    return new ExternalDirectoryBuilder(
         ValuationDate,
         ConfigurationManager.AppSettings["NRRDATE_DIR"],
         ConfigurationManager.AppSettings["NRRDATE_PRICES_FILENAME"]);
}

我的问题是我需要配置文件中的值。现在,所有与配置相关的请求都限制在我的 Ninject 模块中。

如果我使用 Ninject Factory 方法来创建IDirectoryBuilders,我认为我需要将ConfigurationManager 相关调用分散在我的代码库中。

如果我使用 Ninject Provider 方法,我将需要IDirectoryBuilders 的所有实现的提供者,并更新我的构造函数和IDataConnection. 我的代码现在看起来也像(不是很干,与我目前的方法相似)。

Bind<IDirectoryBuilder>().ToProvider<DefaultDirectoryBuilderProvider>()
    .WhenParentNamed("EDDatePricesXLConnection")
    .WithConstructorArgument("baseDir", "someConfigValue")
    .WithConstructorArgument("fileName", "someOtherConfigValue");

我的代码目前有一个非常一致的依赖链(使用 NamedArguments):ICalculator-> IDataReader-> IDataConnection-> IDirectoryBuilder- 这让我相信必须有某种方法可以重复创建这个链,而不必重复设置代码 - 我可以似乎没有弄清楚。还有一个额外的限制,因为我经常需要相同依赖链的两个实例——唯一的区别是不同的配置值。

4

2 回答 2

0

根据鲁本的评论,这是我目前的解决方案。我做的一件事是使用约定的概念,因此检索我的配置设置变得更容易。这过滤了大多数使用 NinjectNamed参数的其他代码。

public void Load(){
    BindDependencies<IDataReader<IList<Price>>, PricesDataReader
      , IDataConnection<IList<PricesCsvRecord>>, PricesXLConnection
      , DefaultDirectoryBuilder>
      ("ValDatePrices");

    BindDependencies<IDataReader<IList<Price>>, PricesDataReader
      , IDataConnection<IList<PricesCsvRecord>>, PricesXLConnection
      , DefaultDirectoryBuilder>
      ("EDDatePrices");
       // etc etc 
}

 public void BindDependencies<
     TReaderBase, TReaderImpl,
     TDataConnectionBase, TDataConnectionImpl,
     TDirectoryBuilderFactoryImpl>
     (string baseName)
         where TReaderImpl : TReaderBase
         where TDataConnectionImpl : TDataConnectionBase
 {
     DirectoryBuilderInfo dirInfor = GetSettings(baseName);

     Bind<TReaderBase>()
           .To(typeof(TReaderImpl))
           .Named(baseName);
     Bind<TDataConnectionBase>().To(typeof(TDataConnectionImpl))
            .WhenParentNamed(baseName)
            .Named(baseName + "XLConnection");
     Func<IDirectoryBuilder> directoryBuilder = CreateDirectoryBuilderFunc<TDirectoryBuilderFactoryImpl>(dirInfor);

     Bind<IDirectoryBuilder>()
            .ToMethod(d => directoryBuilder())
            .WhenParentNamed(baseName + "XLConnection");
    }

private Func<IDirectoryBuilder> CreateDirectoryBuilderFunc<TDirectoryBuilderFactoryImpl>(DirectoryBuilderInfo dirInfor)
{
    Func<IDirectoryBuilder> directoryBuilder = 
         () => CreateDefaultDirectoryBuilder(dirInfor.BaseDirectory, dirInfor.FileName);

    if (typeof(TDirectoryBuilderFactoryImpl) == typeof(RiskDirectoryBuilderFactory))
    {
        directoryBuilder = 
         () => CreateRiskDirectoryBuilder(ValuationDate, dirInfor.BaseDirectory, dirInfor.FileName);
    }
    return directoryBuilder;
}

private DirectoryBuilderInfo GetSettings(string baseName)
{
    var settingsName = baseName.ToUpperInvariant();
    return new DirectoryBuilderInfo()
    {
        BaseDirectory = ConfigurationManager.AppSettings[settingsName + "_DIR"],
        FileName = ConfigurationManager.AppSettings[settingsName + "_FILENAME"]
     };
 }
于 2012-05-03T10:20:27.057 回答
0

没有理由依赖任何 Ninject 特定技术(尽管在某些情况下,提供者可能是合适的(此处提供者示例)。

简单的答案是您为绑定表达式的前一个组件返回的内容创建扩展方法(类似问题)。


在重新阅读您的问题时,我建议您查看Ninject.Extensions.Conventions,它有很多Bind扩展方法,可以让您Bind按照您建议的方式进行操作。如果不是,我建议您评论并识别您认为它没有解决的位。

于 2012-04-27T07:41:51.727 回答