我试图找出最好的(最好的)方法将参数传递给自动解析参数的子对象的构造函数.
为什么?
因为我有一个程序,几乎所有的计算都针对相同的“生产”数据库(在配置文件中定义,所以不需要参数).但是现在需要在这个数据库的“副本”上运行一些工作.因此需要不同的连接字符串.
连接字符串可以在构造函数上提供,并且在编译时是未知的.问题是我不能(或者根本不知道如何)访问这个构造函数,因为它深埋在生成的项目中.
请考虑以下(简化)代码段:
public class Example
{
protected readonly IGenerateSomethingFactory Factory;
public Example(IGenerateSomethingFactory factory)
{
Factory = factory;
}
public Task DoSomething(string ConnectionString, string a, string b)
{
//needs to run somehow using the supplied connection string ...
return Task.Run(() => Factory.CreateSomething().Execute(a, b));
//= Task.Run(() => new Something(new UnitOfWork(new DataContext(ConnectionString))).Execute(a, b));
}
public Task DoSomething(string a, string b)
{
//needs to run using the default connection string (defined in config)
return Task.Run(() => Factory.CreateSomething().Execute(a, b));
}
}
问题出现在第一个DoSomething(…)函数中.
注意:Castle Windsor安装程序如下所示:
public class Installer : IWindsorInstaller
{
public void Install(IWindsorContainer container, IConfigurationStore store)
{
container.AddFacility<TypedFactoryFacility>();
container.Register(Component.For<IGenerateSomethingFactory>().AsFactory());
container.Register(Classes.FromThisAssembly().InNamespace("x").WithService.FirstInterface());
}
}
我正在寻找一个解决方案:
>是线程安全的
>很容易理解(如果不容易想到的话)
>允许重构(所以没有像“conString”这样的命名参数)
>不需要更改其他不相关的代码
(即设置属性公共…)
>不会调用new或container.Resolve<>()
我一直在寻找选择处理程序,但还没找到我想要的东西.
PS:我正在使用Castle Windsor 3.3.0
PPS:这是我的第一个问题,我可以提供更多示例代码,但我认为我应该限制到最低限度……所以如果我需要,请告诉我.
最佳答案 从您的示例中,您需要做的就是在
typed factory’s CreateSomething方法中添加一个参数:
public interface IGenerateSomethingFactory
{
ISomething CreateSomething(string connectionString);
}
然后将其作为参数添加到您的ISomething实现中:
public class Something : ISomething
{
public Something(string connectionString)
{
}
}
注意CreateSomething和Something的构造函数的参数是如何命名的. This is the default behavior for parameter matching.
现在,您只需在调用DoSomething时传递值:
public Task DoSomething(string ConnectionString, string a, string b)
{
return Task.Run(() => Factory.CreateSomething(ConnectionString).Execute(a, b));
}
根据您添加的代码,您尝试做的事情不是立即可行的.简而言之,您拥有此分辨率层次结构:
> IGenerateSomethingFactory.Create(string constring)
> Something.ctor(IUnitOfWork uow)
> UnitOfWork.ctor(IDataContext上下文)
> DataContext.ctor(string constring)
您正在尝试将参数从Create调用传递到DataContext的构造函数.
有关启用此功能的方法,请参阅my answer(我自己的问题).我这样做是通过更改Windsor的默认行为将工厂创建参数向下传递给所有正在解析的对象而不是第一个.
首先,创建此类以更改此行为:
public class DefaultDependencyResolverInheritContext : DefaultDependencyResolver
{
protected override CreationContext RebuildContextForParameter(CreationContext current, Type parameterType)
{
if (parameterType.ContainsGenericParameters)
{
return current;
}
return new CreationContext(parameterType, current, true);
}
}
然后在创建容器时提供它:
var kernel = new DefaultKernel(
new DefaultDependencyResolverInheritContext(),
new NotSupportedProxyFactory());
var container = new WindsorContainer(kernel, new DefaultComponentInstaller());
而已.现在,当您调用Create方法时,constring参数将传递给所有正在注册的对象.显然,如果您的参数名称相同,这可能会导致问题!在您的情况下,这是一个“环境”参数(我的术语),因此您可以记录此行为/参数名称并将其称为一天.
我希望看到另一种方法,除了为所有中间类型创建工厂.