当在写 Controller 的时候,你应该遵守 单一职责,也就意味着你的 Controller 只需做一件事情,换句话说,只有一个因素或者唯一一个因素能让你修改 Controller 中的代码,如果有点懵的话,考虑下面的代码片段,它将 数据访问代码 糅进了 Controller 。
publicclassAuthorController:Controller{privateAuthorContext dataContext =newAuthorContext();publicActionResultIndex(intauthorId){ var authors = dataContext.Authors .OrderByDescending(x=>x.JoiningDate) .Where(x=>x.AuthorId == authorId) .ToList();returnView(authors); }//Other action methods}
请注意上面的代码在 Action 中使用了 dataContext 从数据库读取数据,这就违反了单一职责原则,并直接导致了 Controller 的臃肿。
假如后续你需要修改 数据访问层 代码,可能基于更好的性能或者你能想到的原因,这时候只能被迫在 Controller 中修改,举个例子吧:假如你想把上面的 EF 改成 Dapper 去访问底层的 Database,更好的做法应该是单独拎出来一个 repository 类来操控 数据访问 相关的代码,下面是更新后的 AuthorController。
publicclassAuthorController:Controller{privateAuthorRepository authorRepository =newAuthorRepository();publicActionResultIndex(intauthorId){ var authors = authorRepository.GetAuthor(authorId);returnView(authors); }//Other action methods}
现在 AuthorController 看起来是不是精简多了,上面的代码是不是就是最佳实践呢? 不完全是,为什么这么说呢?上面这种写法导致 Controller 变成了 数据访问组件,取出数据后必然少不了一些业务逻辑处理,这就让 Controller 违反了 单一职责,对吧,更通用的做法应该是将 数据访问逻辑 封装在一个 service 层,下面是优化之后的 AuthorController 类。
publicclassAuthorController:Controller{privateAuthorService authorService =newAuthorService();publicActionResultIndex(intauthorId){ var authors = authorService.GetAuthor(authorId);returnView(authors); }//Other action methods}
再看一下 AuthorService 类,可以看到它利用了 AuthorRepository 去做 CURD 操作。
publicclassAuthorService{privateAuthorRepository authorRepository =newAuthorRepository();publicAuthorGetAuthor(intauthorId){returnauthorRepository.GetAuthor(authorId); }//Other methods}
避免写大量代码做对象之间映射
在 DDD 开发中,经常会存在 DTO 和 Domain 对象,在数据 Input 和 Output 的过程中会存在这两个对象之间的 mapping,按照普通的写法大概就是这样的。
publicIActionResultGetAuthor(intauthorId){ var author = authorService.GetAuthor(authorId); var authorDTO =newAuthorDTO(); authorDTO.AuthorId = author.AuthorId; authorDTO.FirstName = author.FirstName; authorDTO.LastName = author.LastName; authorDTO.JoiningDate = author.JoiningDate;//Other code......}
可以看到,这种一一映射的写法让 Controller 即时膨胀,同时也让 Controller 增加了额外的功能,那如何把这种 模板式 代码规避掉呢?可以使用专业的 对象映射框架 AutoMapper 去解决,下面的代码展示了如何做 AutoMapper 的配置。
publicclassAutoMapping{publicstaticvoidInitialize(){ Mapper.Initialize(cfg => { cfg.CreateMap();//Other code }); }}
接下来可以在 Global.asax 中调用 Initialize() 初始化,如下代码所示:
protectedvoidApplication_Start(){ AutoMapping.Initialize(); }
最后,可以将 mapping 逻辑放在 service 层中,请注意下面的代码是如何使用 AutoMapper 实现两个不兼容对象之间的映射。
publicclassAuthorService{privateAuthorRepository authorRepository =newAuthorRepository();publicAuthorDTOGetAuthor(intauthorId){ var author = authorRepository.GetAuthor(authorId);returnMapper.Map(author); }//Other methods}
避免在 Controller 中写业务逻辑
尽量避免在 Controller 中写 业务逻辑 或者 验证逻辑, Controller 中应该仅仅是接收一个请求,然后被下一个 action 执行,别无其它,回到刚才的问题,这两种逻辑该怎么处理呢?
业务逻辑
这些逻辑可以封装 XXXService 类中,比如之前创建的 AuthorService。
验证逻辑
这些逻辑可以用 AOP 的操作手法,比如将其塞入到 Request Pipeline 中处理。
使用依赖注入而不是硬组合
推荐在 Controller 中使用依赖注入的方式来实现对象之间的管理,依赖注入是 控制反转 的一个子集,它通过外部注入对象之间的依赖从而解决内部对象之间的依赖,很拗口是吧!
一旦你用了依赖注入方式,就不需要关心对象是怎么实例化的,怎么初始化的,下面的代码展示了如何在 AuthorController 下的构造函数中实现 IAuthorService 对象的注入。
publicclassAuthorController:Controller{privateIAuthorService authorService =newAuthorService();publicAuthorController(IAuthorService authorService){this.authorService = authorService; }// Action methods}
使用 action filer 消除 Controller 中的重复代码
可以利用 action filter 在 Request pipeline 这个管道的某些点上安插一些你的自定义代码,举个例子,可以使用 ActionFilter 在 Action 的执行前后安插一些自定义代码,而不是将这些业务逻辑放到 Controller 中,让 Controller 不必要的膨胀,下面的代码展示了如何去实现。
[ValidateModelState][HttpPost]publicActionResultCreate(AuthorRequest request){ AuthorService authorService =newAuthorService(); authorService.Save(request);returnRedirectToAction("Home");}
USB Microphone https://www.soft-voice.com/
Wooden Speakers https://www.zeshuiplatform.com/
亚马逊测评 www.yisuping.cn
深圳网站建设www.sz886.com