MVC架构的职责划分原则,对我颇有启发,分享一下

博主 负责一个项目,用了 Yii Framework 的 MVC 框架,刚开始自觉得结构很稳健。数据库

可是随着对业务逻辑理解的深刻,才开始意识到问题的严重。编程

我错误地理解了 MVC 中的 Controller,想固然地根据以往的经验,把全部的业务逻辑都放在 Controlleraction 中去实现。设计模式

因而,每个 Controller代码都上千行,愈来愈臃肿session

最后,我下定决心重构代码,起源是一个对外开放 API 接口的需求。架构

按照如今的架构,代码基本没法复用,我须要把不少功能再重复写一遍,这实在是没法接受。app

面向对象编程不只仅是课本上的名词啊!框架

真正开始实践才发现,要有面向对象意识,有全局观,是多么可贵的一件事情。yii

1 、到底什么是 MVC

模型-视图-控制器(MVC)是一种设计框架(设计模式)jsp

MVC 的目标将业务逻辑从用户界面的考虑中分离编程语言

这样,开发者就能够更容易地改变每一部分而不会影响其余。

在 MVC 中,

  • Model 表明数据和业务规则
  • View 包含了用户界面元素,例如文本,表单等;
  • Controller 则管理模型和视图中的通讯

MVC 在各类编程语言中均有实现,例如 J2EE 应用开发中,

View 可能由 jsp 实现;Controller 是一个 servlet,如今通常用 Struts 实现;Model 则是由一个实体 Bean 来实现。

2 我遇到了什么问题

Yii Framework 是一个流行的 PHP 框架,它借鉴了 Ruby on Rails 的 ActiveRecord(AR) 概念。

数据库中的每个 table 均可以用 AR 类来方便地进行增删改查操做。

它把 AR 当作 Model,并推荐放在一个名为 models 的目录下面。

因而,我在自动生成表对应的 AR 以后,便望文生义想固然地认为已经拥有了 Model 层。

其实,AR只不过是 DAO (数据访问层),并非 Model 层

咱们的业务几乎全放在了 Controller 里:对用户提交上来的表单进行各类逻辑判断,进行计算,实例化 AR 对数据进行存储……

由于一个 Controller 中会有多个 action,每一个 action 都有这样的业务处理。

最后,我发现个人 Controller 代码已经超过了 1000 行。

忽然有一天,leader 说,咱们这个系统要开放 API 给现有的旧系统调用,要给第三方接口。

第三方只是要给定一个参数,本系统给出个结果值而已,这其中的业务处理它是不关心的。

坏就坏在这里,Controller 已经实现了那些业务,但它是接受表单提交的,怎样可以也接受 SOAP 的 xml 文档呢?

Controller 和套套同样,应该越薄越好。

它的职责应该只是接受用户的输入,而后马上转发给别的类来处理

这样 Controller 只负责提供不一样的接口,咱们才能算是将业务逻辑分离出去,而分离出去的业务也很容易进行重用。

分离出来的这部分业务由谁来处理呢?答案应该是 Model

3 View的职责

View部分比较明确,就是负责显示。

一切与显示界面无关的东西,都不该该出如今view里面。

所以,View 中通常不该该出现复杂的判断语句,以及复杂的运算过程。

能够有简单的循环语句、格式化语句。好比,博客首页的文字列表就是一种循环。

对于PHP的Web应用而言,HTML是View中的主要内容

View应该从不调用Model的写方法

也就是说,View只从Model中读取数据,但不改写Model。

因此咱们说,View和Model是老死不相往来的。

并且,View中不直接访问$_GET$_POST,应该由Controller传递给View。

此外,View通常没有任何准备数据处理的内容,如查询数据库等。

这些通常是放在Controller里面,并以变量的形式传给视图。

也就是说,视图里面要用到的数据,就是一个变量

4 Model的职责

对于Model而言,最主要就是保存和输出信息

好比,Post类必然有一个用于保存博客文章标题的title属性,必然有一个删除的操做,这都是Model的内容。

数据、行为、方法是Model的主要内容

实际工做中,Model是MVC中代码量最大

Model是逻辑最复杂的地方,由于应用的业务逻辑也要在这里表示。

注意将Model与Controller区分开。

Model是处理业务方面的逻辑,Controller只是简单的协调Model和View之间的关系。

只要是与业务有关的,就该放在Model里面。

数据校验、public常量和变量,都应该放在model层,

也就是说,有可能被重复使用的属性或方法,都应该放在model层,一次定义,处处使用。

Model不该该访问request、session以及其余环境数据,这些应该由Controller注入。

好的设计,应该是胖Model,瘦Controller

5 Controller的职责

对于Controller,主要是响应用户请求,决定使用什么视图,须要准备什么数据用来显示

所以,对于request的访问代码,应该放在Controller里面,好比$_GET$_POST等。

Controller应该仅限于获取用户请求数据,不该该对数据有任何操做或预处理,这应该放在 Model 里面。

对于数据的写操做,要调用Model类的方法完成。

对于用户请求的响应,要调用视图渲染。

此外,通常不要有HTML代码等其余表现层的东西,这应该是属于View的内容。

6 启示

Yii Framework 的官方文档中有这么一段:

In a well-designed MVC application, controllers are often very thin, containing probably only a few dozen lines of code; while models are very fat, containing most of the code responsible for representing and manipulating the data.

在一个设计良好的MVC应用程序中,控制器一般很薄,可能只包含几十行代码;而模型很是庞大,包含了大部分负责表示和操做数据的代码。