- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
在使用最新的 StructureMap 版本 (3.1.6.191) 升级我们的解决方案后,我收到了很多过时的警告。这些警告来自 StructureMap,其中 ObjectFactory
类将在未来版本 (4.0+) 中弃用。
我正在使用 WCF,我们希望使用 IInstanceProvider
的实现让 StructureMap Hook 到 WCF 管道中:
public class StructureMapInstanceProvider : IInstanceProvider
这个类使用 ObjectFactory
来获取一个实例,当我的容器没有静态类来解析它时,我们如何获取一个类型的实例?
最佳答案
ObjectFactory
即将消失,因为许多人认为它 anti-pattern从应用程序中访问容器(称为服务定位器模式)。这会将您的代码与容器紧密耦合,并且难以维护配置,因为确定类需要哪些依赖项并不容易。
依赖注入(inject)不同于使用服务定位器。使用依赖注入(inject),对象图在 composition root 中的应用程序开始附近解析。 .一旦创建了应用程序,它就没有对 IoC 容器的引用,因此与它没有紧密耦合。依赖项在类构造函数中明确定义,因此您无需进一步查找即可发现类在注册时需要哪些依赖项。
在运行时,您不可避免地需要创建类的实例。为此,您可以求助于众多 Creational Patterns 之一(其中抽象工厂最常见,IInstanceProvider
实现)或者,您可以 inject a method使用容器来创建这些实例。
我推荐阅读这本书 Dependency Injection in .NET .有一个部分 (7.3) 专门介绍了通过实现 ServiceHost
、ServiceHostFactory
和 IInstanceProvider
将 WCF 与组合根连接起来。
这是使用 StructureMap 的 WCF 组合根的基本示例(尽管我尚未验证它是否有效)。
这是您向容器注册类型的地方。如果愿意,您可以使用多个注册表。
public class MyRegistry : Registry
{
public MyRegistry()
{
// Register all types
this.For<ISomeService>().Use<SomeService>();
}
}
这是我们实例化容器并注册类型映射的地方。
public class MyServiceHostFactory : ServiceHostFactory
{
private readonly IContainer container;
public MyServiceHostFactory()
{
this.container = new Container(r => r.AddRegistry<MyRegistry>());
}
protected override ServiceHost CreateServiceHost(Type serviceType, Uri[] baseAddresses)
{
return new MyServiceHost(this.container, serviceType, baseAddresses);
}
}
这是我们注入(inject)容器的地方。是的,我们至少需要在一个地方这样做。在这里没问题,因为这是插入 WCF 的组合根的所有部分。
public class MyServiceHost : ServiceHost
{
public MyServiceHost(IContainer container, Type serviceType, params Uri[] baseAddresses)
: base(serviceType, baseAddresses)
{
if (container == null)
throw new ArgumentNullException("container");
var contracts = this.ImplementedContracts.Values;
foreach (var c in contracts)
{
c.Behaviors.Add(new MyInstanceProvider(container, serviceType));
}
}
}
我们仍然处于应用程序的组合根中,因此我们可以注入(inject)容器来解析我们的实例。
由这个抽象工厂解析的您的服务不应该有对容器的引用(既不是静态的也不是注入(inject)的)。
public partial class MyInstanceProvider : IInstanceProvider, IContractBehavior
{
private readonly IContainer container;
private readonly Type serviceType;
public MyInstanceProvider(IContainer container, Type serviceType)
{
if (container == null)
throw new ArgumentNullException("container");
this.container = container;
this.serviceType = serviceType;
}
public object GetInstance(InstanceContext instanceContext, Message message)
{
return this.GetInstance(instanceContext);
}
public object GetInstance(InstanceContext instanceContext)
{
return this.container.GetInstance(this.serviceType);
}
public void ReleaseInstance(InstanceContext instanceContext, object instance)
{
// Allow the lifetime management behavior of StructureMap to release dependencies
}
public void ApplyDispatchBehavior(
ContractDescription contractDescription, ServiceEndpoint endpoint,
DispatchRuntime dispatchRuntime)
{
dispatchRuntime.InstanceProvider = this;
}
}
只需将这些内容添加到您的 .svc
文件中即可注册自定义 MyServiceHostFactory
来解析您的 WCF 服务。
<%@ ServiceHost Factory = "MyNamespaceName.MyServiceHostFactory, MyAssemblyName" Service = "MyNamespaceName.MyWcfService" %>
使用的引用资料:
关于c# - 如何使用 StructureMap 摆脱代码中的 ObjectFactory,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32732650/
http://docs.structuremap.net/似乎有使用已弃用成员的非常古老的示例。 有什么地方可以找到最新的 StructureMap doco 吗? 最佳答案 我似乎在以下位置找到了一
我似乎无法弄清楚如何将对象定义为单例并为构造函数定义两个参数。 我可以做/或 .. 只是不能同时做。 例如。 (这不起作用)... ForRequestedType() .TheDefault
有没有办法确定在 StructureMap 中是否配置了特定类型? 如果在 StructureMap 中没有专门配置,我想返回一个泛型类型。 最佳答案 在 v2.6 你想要: IContainer.M
我想配置结构图以使用工厂类创建服务。工厂本身有一个需要填充的依赖项。目前我的注册表类中有以下内容: For().Singleton().Use(() => new DoStuffWebServ
我正在尝试使用 StructureMap 2.6.1 使用基于约定的注册一次注册我的所有存储库。见下面的代码: x.Scan(s => { s.TheCallingAssembly();
在 StructureMap 中你可以声明一个 Forward声明,这将允许注册单个具体实例,以由来自 StructureMap documentation 的多个接口(interface)解析: v
在 StructureMap 2 中,我有这样的事情: For().HybridHttpOrThreadLocalScoped().Use(); 使用 Structure Map 3 时,我应该使用以
如何将 StructureMap 与 OpenRasta 一起使用?我可以使用它来代替内部依赖解析器,还是只能将它与内置 DI 结合使用(即用于我自己的应用程序的依赖项)? 谢谢 最佳答案 Struc
我刚刚开始使用 StructureMap,之前使用过 Spring.Net。我喜欢 DefaultConventionScanner 以及扫描程序集和使用约定优于配置来查找类的能力。但是似乎有一个限制
我有一个通用接口(interface) public interface IDomainDataRepository { T[] GetAll(); } 有一个通用的实现 public cla
我正在尝试从 Structuremap 2.6.4.1 迁移到 3.1.4.143,但无法弄清楚如何处理 HybridHttpOrThreadLocalScoped。我能找到的所有 SO Q/A 似乎
所以我有一种情况,我想注册 n 个特定查找类型的映射。即: x.For().Add(); x.For().Add(); 我想让 SM 将它们的可枚举(或数组)注入(inject)到类的构造函数中: p
以下哪种语法被认为是最佳实践? For().LifecycleIs(new HybridLifecycle()).Use(); For().LifecycleIs(Lifecycles.GetLife
StructureMap 有没有办法用一行或约定来做这种重复映射? For>().Use(); For>().Use(); For>().Use(); For>().U
似乎 NLog 不能对 GetCurrentClassLogger() 使用反射,即使我的 MVC 3 应用程序部署在 中完全信任 IIS7 环境。我使用的是 StructureMap 2.6.1,问
StructureMap 定义了一个接口(interface) IBootStrapper,我看到很多人在他们的 Bootstrap 类中实现了这个接口(interface)。 但是我找不到任何可以说
我们将 Caliburn.Micro 用于我们的 MVVM 框架,将 StructureMap 用于我们的 IoC 容器,并将 MediatR 用于我们的调解器实现。这一切都工作正常,除了注册 Med
使用方法DecorateAllWith用 DynamicProxy 装饰所有实例都实现了一个接口(interface)? 例如: public class ApplicationServiceInte
在 autoFac 中,我可以注册一个接口(interface)的多个实现。当 autofac 实例化我的对象时,所有实例都传递给构造函数。 来自 autofac 的文档:here For examp
我有一个 Repository 的基类。在一个特定的项目中,我有几个这个基类的实现。例如 PersonRepository : Repository EmployerRepository : Repo
我是一名优秀的程序员,十分优秀!