如何将参数传递给待解析类型的 'child' 项的构造函数
How to pass an argument to the constructor of a 'child' item of the to-be-resolved type
我正在尝试找出将参数传递给自动解析参数的子对象的构造函数的最佳(最好)方法。
为什么?
因为我有一个程序几乎所有的计算都是针对同一个 "production" 数据库(在配置文件中定义的,所以不需要参数)。但现在需要 运行 对该数据库的 'copy' 进行一些处理。因此需要不同的连接字符串。
连接字符串可以在构造函数中提供,并且在编译时是未知的。问题是我不能(或不知道如何)简单地访问这个构造函数,因为它深埋在生成的项目中。
考虑以下(简化的)代码片段:
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" 这样的命名参数)
- 不需要更改其他不相关的代码
(即设置属性 public ...)
- 不调用
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 context)
DataContext.ctor(string constring)
您正在尝试将参数从对 Create
的调用传递到 DataContext
的构造函数。
有关启用此功能的方法,请参阅 my answer(针对我自己的问题)。为此,我更改了 Windsor 的默认行为,将工厂创建参数传递给所有正在解析的对象,而不仅仅是第一个对象。
首先,创建此 class 以更改此行为:
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
参数将传递给所有正在注册的对象。 显然,如果您有同名参数,这可能会导致问题!在您的情况下,这是一个 "ambient" 参数(我的术语),因此您可以记录这个 behavior/parameter 命名并收工。
除了为所有中间类型创建工厂之外,我很想看到另一种方法。
我正在尝试找出将参数传递给自动解析参数的子对象的构造函数的最佳(最好)方法。
为什么? 因为我有一个程序几乎所有的计算都是针对同一个 "production" 数据库(在配置文件中定义的,所以不需要参数)。但现在需要 运行 对该数据库的 'copy' 进行一些处理。因此需要不同的连接字符串。
连接字符串可以在构造函数中提供,并且在编译时是未知的。问题是我不能(或不知道如何)简单地访问这个构造函数,因为它深埋在生成的项目中。
考虑以下(简化的)代码片段:
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" 这样的命名参数)
- 不需要更改其他不相关的代码 (即设置属性 public ...)
- 不调用
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 context)
DataContext.ctor(string constring)
您正在尝试将参数从对 Create
的调用传递到 DataContext
的构造函数。
有关启用此功能的方法,请参阅 my answer(针对我自己的问题)。为此,我更改了 Windsor 的默认行为,将工厂创建参数传递给所有正在解析的对象,而不仅仅是第一个对象。
首先,创建此 class 以更改此行为:
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
参数将传递给所有正在注册的对象。 显然,如果您有同名参数,这可能会导致问题!在您的情况下,这是一个 "ambient" 参数(我的术语),因此您可以记录这个 behavior/parameter 命名并收工。
除了为所有中间类型创建工厂之外,我很想看到另一种方法。