Page 类和演示者都不必直接处理管理其任何依赖项的构造或生命周期 - 这一切都应该由您的容器处理。由于构造函数注入不适用于 WebForms,因此您需要将任何需要的依赖项公开为类的属性。例如,您可以将班级更改为:
public partial class _Default : System.Web.UI.Page
{
protected void Page_Load(object sender, EventArgs e)
{
}
public _DefaultPresenter Presenter { get; set; }
}
该页面不需要对存储库的任何引用,因为它将被注入到演示者中。
此答案的其余部分特定于 StructureMap - 其他容器的详细信息可能有所不同。
要启用 setter 注入,您需要告诉 StructureMap 要填充哪些属性。一种方法是将 [SetterProperty] 属性应用于属性本身。但是,在您的类中包含 StructureMap 详细信息可能会让人觉得有点侵入性。另一种方法是配置 StructureMap 以便它知道要注入哪些属性类型。例如:
protected void Application_Start(object sender, EventArgs e)
{
ObjectFactory.Initialize(x =>
{
x.Scan(scan =>
{
scan.TheCallingAssembly();
scan.WithDefaultConventions();
});
x.ForRequestedType<IRepository>().TheDefaultIsConcreteType<Repository>().CacheBy(InstanceScope.Hybrid);
x.SetAllProperties(set => set.WithAnyTypeFromNamespaceContainingType<IRepository>());
});
}
SetAllProperties 方法允许您告诉 StructureMap 如何识别它应该填充的属性。在这种情况下,我告诉 StructureMap 注入所有演示者(假设它们都在同一个命名空间中)。
您仍然需要对每个请求执行 setter 注入。使用 StructureMap,您可以使用 BuildUp() 方法将依赖项注入现有实例。您可以在每个页面或页面基类的 Init 或 Load 事件中执行此操作,但同样,这让人感觉很侵入。要将容器完全排除在页面类之外,您可以使用应用程序的 PreRequestHandlerExecute 事件(在 global.asax 或 IHttpModule 中):
protected void Application_PreRequestHandlerExecute(object sender, EventArgs e)
{
var application = (HttpApplication)sender;
var page = application.Context.CurrentHandler as Page;
if (page == null) return;
ObjectFactory.BuildUp(page);
}
最后,如果你想显式地处理你的 IRepository,你可以在 EndRequest 事件中处理它:
protected void Application_EndRequest(object sender, EventArgs e)
{
var disposable = ObjectFactory.GetInstance<IRepository>() as IDisposable;
if (disposable != null) disposable.Dispose();
}
请注意,这可以正常工作,因为在初始化时我们告诉 StructureMap 通过 Hybrid 缓存 IRepository,这意味着“为每个 HTTP 请求(或线程,如果不在网站中运行)给我相同的实例”。当您在 EndRequest 中检索 IRepository 时,您将收到在整个请求中使用的同一个,您可以将其丢弃。