在计算机科学中,我们被教导说每种方法都应该做一件事,而且只做一件事。我有点困惑,我们看到像下面这样的 MVC 操作作为良好实践的例子:
[AcceptVerbs(HttpVerbs.Post), Authorize]
public ActionResult Edit(int id, FormCollection collection) {
Dinner dinner = dinnerRepository.GetDinner(id);
if (!dinner.IsHostedBy(User.Identity.Name))
return View("InvalidOwner");
try {
UpdateModel(dinner);
dinnerRepository.Save();
return RedirectToAction("Details", new { id=dinner.DinnerID });
}
catch {
ModelState.AddModelErrors(dinner.GetRuleViolations());
return View(new DinnerFormViewModel(dinner));
}
}
基本上这段代码提供了很多功能:
- 定义如何访问操作 - 仅发布
- 定义谁可以访问操作 - 授权
- 访问持久化机制 -dinnerRepository
- 访问状态信息 - (User.Identity.Name)
- 将 NameValueCollection 转换为强类型对象 - UpdateModel()
- 为每个指定 3 个可能的 ActionResult 和内容 - InvalidOwner/Details/Edit 视图
对我来说,这似乎是一种方法的太多责任。这也是一个相当简单的操作,即它不处理常见场景,例如:
- 检查业务规则 - “永远不要相信用户输入”
- 导航路径 - 成功保存后始终返回“详细信息”
- 不同的返回类型 - 有人想从网格中调用“编辑”并需要 JsonResult?
- 更好的错误处理 - 如果在 GetDinner(id) 期间无法访问数据库,则 YSOD
- 构建额外的视图数据 - 下拉列表的 SelectLists
更不用说围绕这个单一方法所需的测试量,即模拟/伪造 FormCollection/UserIdentity/Authorization Provider/Repository/等。
我的问题是我们如何避免在控制器动作中塞进这么多东西?
我倾向于认为“意见”是一个很好的概念,尤其是“雷电原则”。虽然我非常尊重参与构建FubuMVC的人及其背后的原因,但我需要一些我现在可以使用的东西。
编辑-好吧,看来我是在追求这样的东西-自以为是的控制器。我需要进一步检查它,因为它适用于 MVC Preview 5,所以我可能需要自己更新它。