对于那些在ASP.NET MVC中创建viewmodels(由类型化视图使用)的用户,您是否希望从viewmodel或控制器类中的服务/存储库中获取数据?
例如,我们开始使viewmodel基本上是DTO,并允许我们的控制器获取数据(严重过于简单的示例假定用户只能更改员工名称):
public class Employeeviewmodel { public String Name; //posted back public int Num; //posted back public IEnumerable<Dependent> Dependents; //static public IEnumerable<Spouse> Spouses; //static } public class EmployeeController() { ... public ActionResult Employee(int empNum) { Models.Employeeviewmodel model = new Models.Employeeviewmodel(); model.Name = _empSvc.FetchEmployee(empNum).Name; model.Num = empNum; model.Dependents = _peopleSvc.FetchDependentsForView(empNum); model.Spouses = _peopleSvc.FetchDependentsForView(empNum); return View(model); } [AcceptVerbs(HttpVerbs.Post)] public ActionResult Employee(Models.Employeeviewmodel model) { if (!_empSvc.ValidateAndSaveName(model.Num,model.Name)) { model.Dependents = _peopleSvc.FetchDependentsForView(model.Num); model.Spouses = _peopleSvc.FetchDependentsForView(model.Num); return View(model); } this.RedirectToAction(c => c.Index()); } }
这一切似乎都很好,直到我们开始创建大量视图(40个字段),有许多下拉列表等.由于屏幕将具有GET和POST操作(如果出现验证错误,则POST返回视图),因此我们将复制代码并使viewmodels的大小应大于此值.
我认为替代方案是通过viewmodel中的服务获取数据.我担心的是,我们可以从viewmodel中收集一些数据,另外还有一些来自Controller的数据(例如,在上面的例子中,Name将从Controller中填充,因为它是一个已发布的值,而Dependents和Spouses将通过一些viewmodel中GetStaticData()函数的类型).
思考?
解决方法
我遇到同样的问题.当代码对于动作方法来说太大时,我开始为每个动作创建类.是的,您将在类中进行一些数据检索,并在控制器方法中进行一些检索.另一种方法是将所有数据检索到类中,但是您将不会真正需要的一半类,它们将被创建以保持一致性,或者在控制器方法中进行所有数据检索,但是再一次,这些方法中的一些将会太复杂了,需要被抽象成类…所以选择你的毒药.我宁愿有一点不一致,并有正确的解决办法.
至于将行为放入viewmodel中,我不这样认为,viewmodel的一个要点是从View中设置和提取值的一个薄类.
有一些我把转换方法放在viewmodel中的情况.例如,我需要将viewmodel转换为相应的实体,或者我需要使用实体的数据加载viewmodel.
为了回答你的问题,我更喜欢在controller / action方法中检索数据.
通常使用DropDowns,我创建一个下拉列表服务. DropDowns往往是与视图相同的数据.通过服务中的下拉菜单,我可以将其用于其他视图和/或缓存.
根据布局,40个字段可以创建一个混乱的视图.根据数据类型,我会尝试通过某种标签或向导界面将多个视图跨越多个视图.