浅谈PHP-MVC模式及简单实现
何为MVC?
MVC模式(Model–view–controller)是软件工程中的一种软件架构模式,把软件系统分为三个基本部分:模型(Model)、视图(View)和控制器(Controller)。
MVC模式最早由Trygve Reenskaug在1978年提出[1],是施乐帕罗奥多研究中心(Xerox PARC)在20世纪80年代为程序语言Smalltalk发明的一种软件架构。MVC模式的目的是实现一种动态的程序设计,使后续对程序的修改和扩展简化,并且使程序某一部分的重复利用成为可能。除此之外,此模式透过对复杂度的简化,使程序结构更加直观。
模型(Model)
用于封装与应用程序的业务逻辑相关的数据以及对数据的处理方法。“ Model ”有对数据直接访问的权力,例如对数据库的访问。“Model”不依赖“View”和“Controller”,也就是说, Model 不关心它会被如何显示或是如何被操作。但是 Model 中数据的变化一般会通过一种刷新机制被公布。为了实现这种机制,那些用于监视此 Model 的 View 必须事先在此 Model 上注册,从而,View 可以了解在数据 Model 上发生的改变。
视图(View)
能够实现数据有目的的显示(理论上,这不是必需的)。在 View 中一般没有程序上的逻辑。为了实现 View 上的刷新功能,View 需要访问它监视的数据模型(Model),因此应该事先在被它监视的数据那里注册。
控制器(Controller)
起到不同层面间的组织作用,用于控制应用程序的流程。它处理事件并作出响应。“事件”包括用户的行为和数据 Model 上的改变。
接下来我们用代码演绎一下
首先创建目录mvc,下面创建文件目录mvc,创建composer.json
如下
{
"name": "xia/mymvc",
"require": {},
"autoload": {
"psr-4": {
"xia\\app\\": "mvc/"
}
},
"authors": [{
"name": "xia",
"email": "xia@qq.com"
}]
}
运行composer install
(主要是为了获得类的自动加载),创建其它文件,目录结构如下:
Model.php
<?php
// 获取模型数据
namespace xia\app;
use PDO;
class Model
{
public function getData()
{
return (new PDO('mysql:host=localhost;charset=utf8;dbname=phpcn', 'root', '', [PDO::ATTR_ERRMODE => PDO::ERRMODE_WARNING]))->query('SELECT `cou_id`,`title`,`pic`,`info`,`add_time`FROM `mj_course_lists` ORDER BY `cou_id` DESC LIMIT 6')->fetchAll();
}
public function editData()
{
}
}
View.php
<?php
namespace xia\app;
class View
{
// $data渲染到视图层的数据
public function fetch($data)
{
$table = '<table border="1" cellspacing="0" cellpadding="5" align="center">';
$table .= '<caption>课程信息表</caption>';
$table .= '
<tr align="center">
<td>编号</td>
<td>名称</td>
<td>封面</td>
<td>课程简介</td>
<td>创建时间</td>
<td>操作</td>
</tr>
';
foreach ($data as $list) {
$table .= '<tr>';
$table .= '<td>' . $list['cou_id'] . '</td>';
$table .= '<td>' . $list['title'] . '</td>';
$table .= "<td><img style='width:100px' src='{$list['pic']}' alt='课程封面' ></td>";
$table .= '<td>' . $list['info'] . '</td>';
$table .= '<td>' . date("Y-m-d H:m:s", $list['add_time']) . '</td>';
$table .= '<td><button>删除</button> <button>编辑</button> </td>';
$table .= '</tr>';
}
$table .= '</table>';
return $table;
}
}
Controller.php
<?php
namespace xia\app;
// 类的自动加载
require __DIR__ . DIRECTORY_SEPARATOR . '..' . DIRECTORY_SEPARATOR . 'vendor' . DIRECTORY_SEPARATOR . 'autoload.php';
// 中间层 controller 协调 m v
class Controller
{
protected $container;
// 将外部依赖的对象 model view 在构造器中注入进来 完成外部对象在多个方法中共享
function __construct($container)
{
$this->container = $container;
}
function index()
{
// 获取数据
$data = $this->container->make('model')->getData();
// 渲染数据到视图层
return $this->container->make('view')->fetch($data);
}
function edit()
{
$this->container->make('model')->editData();
}
}
$container = new Container;
// 绑定一个闭包到服务容器中 我们使用$model对象时才去实例化
$container->bind('model', function () {
return new Model;
});
$container->bind('view', function () {
return new View;
});
$c2 = new Controller($container);
echo $c2->index();
依赖注入(dependency injection)
依赖”是指可被方法调用的事物。依赖注入形式下,调用方不再直接使用“依赖”,取而代之是“注入” 。“注入”是指将“依赖”传递给调用方的过程。在“注入”之后,调用方才会调用该“依赖”。
在PHP中,如果当前类所依赖的外部对象过多,名称很长 将依赖的所有外部对象放到一个容器中统一管理,并且还可以起别名
服务容器
服务容器是一个用于管理类依赖和执行依赖注入的工具,是一个自动产生类、对象的工厂。
Container.php
<?php
namespace xia\app;
use Closure;
// 服务容器 一个自动生产类 /对象的工厂
class Container
{
// 对象容器
protected $instances = [];
/**
* 绑定一个类标识、闭包、实例到容器
* @access public
* @param string|array $abstract 类标识或者接口的别名 alias
* @param mixed $concrete 要绑定的类、闭包或者实例
*
*/
// 往容器数组中绑定对象
public function bind($abstract, Closure $concrete)
{
$this->instances[$abstract] = $concrete;
}
public function make($abstract, $params = [])
{
return call_user_func_array($this->instances[$abstract], $params);
}
}