这是一个棘手的问题 - 当你想在ASP.NET Web窗体中使用动态控件时,你必须选择两个恶意中较小的一个。最终,这归结于您想要达成的妥协方案:演示者在视图中担忧或担心演示者的担忧。
当我在网页表单的工作,我通常更喜欢后者 - 我接受了主持人是联系在一起的ASP.NET页面的生命周期,并创建一个InitializeView
方法做任何有必要创建动态控件:
// Presenter
// This could also be parameterless if you prefer that idiom but
// then the view needs a SelectedState property that serves up values
// straight from the Form collection, and it won't be obvious why.
public void InitializeView(string selectedState) {
if (selectedState != null) {
view.Counties = dataLayer.GetCounties(selectedState);
}
}
// View
protected void Page_Init(object sender, EventArgs args) {
presenter.InitializeView(Request.Form["StateList"]);
// ... build counties drop-down ...
}
当然,这会将演示者的语义与ASP.NET生命周期联系起来,并遮蔽了该方法中正在发生的事情。您可以通过给予InitializeView
一个更具描述性的名称(如ProcessSelectedState
)来缓解这种情况,但除非方法名称引用页面生命周期,否则它永远不会很明显,为什么您不只是让其他模型的县(在presenter.LoadModel
或其他你可以称它)。
我可以看到替代是如何吸引人:
protected void Page_Init(object sender, EventArgs args) {
if (Request.Form["StateList"] != null) {
List<string> counties = presenter.GetCounties(Request.Form["StateList"]);
// ... build counties drop-down ...
}
}
的Presenter
的语义是非常明确 - 这是很容易理解什么GetCounties
不和它没有任何与页面生命周期。但是在你看来,你有可测试的东西,这是一件令人失望的事情,而这通常比让我的演示者不理解他们的视图引擎更重要。
另一种替代方法是在页面初始化期间只加载您的整个模型。您的服务器控制值将不可用,因此您必须从Request.Form
中获取任何这些值。这虽然不是惯用的ASP.NET经典 - 但它可能会让人过度困惑,因为大多数Web表单开发人员习惯于直接从Web控件获取值,而不是直接从POST数据获取值。
我还不清楚为什么主持人需要知道页面的生命周期。主持人是不是仅仅为视图提供一个集合?服务器控件是不是在视图中创建和渲染的? – BlackICE 2010-04-05 14:38:25