具有使用相同接口(StructureMap 或任何其他 DI 框架)的多个数据库的 IOC
我们一直在尝试 StructureMap,但我很难掌握如何处理单个接口有多个实现的情况。下面的代码显示了一个示例,其中我们有两个数据库,都可以从单个服务访问。
public class SomeController : Controller
{
private ISomeService _service;
private IClientRepository _repository;
protected IContext _masterContext;
protected IContext _clientContext;
public SomeController(ISomeService service, ISomeRepository repository
, IContext masterCon, IContext clientCon)
{
_service = service;
_repository = repository;
_masterContext = masterCon;
_clientContext = clientCon;
}
}
public class SomeService : ISomeService
{
private IContext _masterContext;
private IContext _clientContext;
public SomeService(IContext masterContext, IContext clientContext)
{
masterContext = _masterContext;
clientContext = _clientContext;
}
}
public class ClientRepository : IClientRepository
{
private IContext _clientContext;
public ClientRepository(IContext clientContext)
{
_clientContext = clientContext;
}
}
public class MasterContext : IContext
{
public MasterContext(String connString)
//<snip, snip> implement 3rd party data context
}
public class ClientContext : IContext
{
public ClientContext(String connString)
//<snip, snip> implement 3rd party data context
}
当我们只有一个上下文(数据库)时,StructureMap 工作得很好,但是我如何告诉它如何解析第二个呢?注意:在大多数情况下,我们不会有一个服务处理 2 个数据库(但可能有一个控制器处理 2 个连接,即 2 个存储库访问 2 个不同的数据库),但它似乎仍然没有让事情变得更容易。
我已经准备好放弃使用 IoC 框架并回到穷人的 DI 了。
We've been experimenting with StructureMap, and I'm having trouble grasping how to handle situations where a single interface has multiple implementations. The code below shows an example where we have two databases that are both accessible from a single service.
public class SomeController : Controller
{
private ISomeService _service;
private IClientRepository _repository;
protected IContext _masterContext;
protected IContext _clientContext;
public SomeController(ISomeService service, ISomeRepository repository
, IContext masterCon, IContext clientCon)
{
_service = service;
_repository = repository;
_masterContext = masterCon;
_clientContext = clientCon;
}
}
public class SomeService : ISomeService
{
private IContext _masterContext;
private IContext _clientContext;
public SomeService(IContext masterContext, IContext clientContext)
{
masterContext = _masterContext;
clientContext = _clientContext;
}
}
public class ClientRepository : IClientRepository
{
private IContext _clientContext;
public ClientRepository(IContext clientContext)
{
_clientContext = clientContext;
}
}
public class MasterContext : IContext
{
public MasterContext(String connString)
//<snip, snip> implement 3rd party data context
}
public class ClientContext : IContext
{
public ClientContext(String connString)
//<snip, snip> implement 3rd party data context
}
StructureMap worked GREAT when we had a single context (database), but how do I tell it how to resolve the 2nd? Note: in most situations we wouldn't have a service handling 2 databases (but may have a controller handling 2 connections, i.e. 2 repositories accessing 2 different databases), but it still doesn't seem to make it easier.
I'm half ready to just give up on using an IoC framework and go back to poor man's DI.
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(4)
是否不可能有一个
IClientContext
和一个IMasterContext
,可能继承自IContext
。我的感觉是,代码将执行两种截然不同的操作之一,具体取决于您是与“主”数据库还是“客户端”数据库通信。Is it not possible to have an
IClientContext
and anIMasterContext
, possibly inheriting fromIContext
. My feeling is that the code would be doing one of two very different things depending on whether you were talking to the 'Master' or 'Client' database.在 Unity 中,您可以进行命名注册,从而允许您为给定接口有效注册多个类。所以你可以这样做(用心打字,如果有兴趣,请检查实际的 Unity 文档):
然后 SomeService 的构造函数将是:
缺点是,通过这种方式,你的服务类不再独立于所使用的 DI 框架,而是依赖于可能还可以的项目。
In Unity you can have named registrations, allowing you to effectively register more than a class for a given interface. So you could do (typing by heart, check the actual Unity documentation if interested):
and then the constructor for SomeService would be:
The drawback is that in this way your service class is no longer independent of the DI framework used, but depending on the project that may be ok.
如果您依赖 StructureMap 自动解决依赖关系,这可能会有点困难。第一个解决方案(也是我会犯的错误)是利用像理查德在他的答案中提到的标记接口,然后注册它们。然后,您可以明确指定是否需要客户端上下文或主上下文。
第二种方法是使用命名注册,然后显式指定构造函数参数。
不是特别好,但它可以完成工作,最终如果它只在一两个地方,它可能还可以。
如果您想对命名空间/程序集进行不同的解析,您可以尝试如下操作:-
ParentType 上的谓词可以引用程序集(或您真正想要的任何内容)
This can be a little difficult if you're relying on StructureMap to resolve the dependencies automatically. The first solution (and what I'd err towards) is to make use of marker interfaces like Richard mentions in his answer then just register them. You can then explicitly specify whether you want your client or master context there.
The second way is to make use of named registrations, then specify the constructor params explicitly.
Not particularly nice but it does the job and ultimately if it's only in one or two places it might be OK.
If you want to resolve differently on namespace / assembly you could try something like this:-
Where the predicate on ParentType can refer to Assembly (or whatever you want really)
如果有人遇到这个问题,您可以使用工厂模式来实现。
服务扩展
工厂类
Sql类
用法
输出
依赖项:
In case someone stumble in this problem, you can achieve it using factory pattern.
Service extension
Factory class
Sql classes
Usage
Output
Dependencies: