mvc 自己封装一个框架
MVC模式(Model-View-Controller)是软件工程中的一种软件架构模式。
MVC把软件系统分为三个基本部分:模型(Model)、视图(View)和控制器(Controller)。
PHP中MVC模式也称Web MVC,从上世纪70年代进化而来。
MVC的目的是实现一种动态的程序设计,便于后续对程序的修改和扩展简化,并且使程序某一部分的重复利用成为可能。
除此之外,此模式通过对复杂度的简化,使程序结构更加直观。
MVC各部分的职能:
模型Model – 管理大部分的业务逻辑和所有的数据库逻辑。模型提供了连接和操作数据库的抽象层。
控制器Controller - 负责响应用户请求、准备数据,以及决定如何展示数据。
视图View – 负责渲染数据,通过HTML方式呈现给用户。
一个典型的Web MVC流程:
Controller截获用户发出的请求;
Controller调用Model完成状态的读写操作;
Controller把数据传递给View;
View渲染最终结果并呈献给用户。
在开始开发前,让我们先来把项目建立好。
假设我们建立的项目为 mvc,MVC的框架命名为 mvc,那么接下来,第一步要把目录结构设置好。
├─application 应用目录
│ ├─controllers 控制器目录
│ ├─models 模块目录
│ ├─views 视图目录
├─config 配置文件目录
├─vendor 框架核心目录
├─static 静态文件目录
├─index.php 入口文件
代码规范
在目录设置好以后,我们接下来规定代码的规范:
- MySQL的表名需小写或小写加下划线,如:item,car_orders。
- 模块名(Models)需用大驼峰命名法,即首字母大写,并在名称后添加Model,如:ItemModel,CarModel。
- 控制器(Controllers)需用大驼峰命名法,即首字母大写,并在名称后添加Controller,如:ItemController,CarController。
- 方法名(Action)需用小驼峰命名法,即首字母小写,如:index,indexPost。
- 视图(Views)部署结构为控制器名/行为名,如:item/view.php,car/buy.php。
上述规则是为了程序能更好地相互调用。
重定向
重定向的目的有两个:设置根目录为project所在位置,以及将所有请求都发送给 index.php 文件。
如果是Apache服务器,在 project 目录下新建一个 .htaccess 文件,内容为:
<IfModule mod_rewrite.c>
# 打开Rerite功能
RewriteEngine On
# 如果请求的是真实存在的文件或目录,直接访问
RewriteCond %{REQUEST_FILENAME} !-f
RewriteCond %{REQUEST_FILENAME} !-d
# 如果访问的文件或目录不是真事存在,分发请求至 index.php
RewriteRule . index.php
</IfModule>
如果是Nginx服务器,修改配置文件,在server块中加入如下的重定向:
location / {
# 重新向所有非真是存在的请求到index.php
try_files $uri $uri/ /index.php$args;
}
这样做的主要原因是:
(1)静态文件能直接访问。
如果文件或者目录真实存在,则直接访问存在的文件/目录。
比如,静态文件static/css/main.css真实存在,就可以直接访问它。
(2)程序有单一的入口。
这种情况是请求地址不是真实存在的文件或目录,这样请求就会传到 index.php 上。
例如,访问地址:localhost/item/view/1,在文件系统中并不存在这样的文件或目录。
那么,Apache或Nginx服务器会把请求发给index.php,并且把域名之后的字符串赋值给REQUEST_URI变量。
这样在PHP中用$_SERVER['REQUEST_URI']就能拿到/item/view/1;
(3)可以用来生成美化的URL,利于SEO。
3.4 入口文件
接下来,在 project 目录下新建 index.php 入口文件,文件内容为:
<?php
// 应用目录为当前目录
define('APP_PATH', __DIR__ . '/');
// 开启调试模式
define('APP_DEBUG', true);
// 加载框架文件
require(APP_PATH . 'fastphp/Fastphp.php');
// 加载配置文件
$config = require(APP_PATH . 'config/config.php');
// 实例化框架类
(new Fastphp($config))->run();
注意,上面的PHP代码中,并没有添加PHP结束符号?>。
这么做的主要原因是:
对于只有 PHP 代码的文件,最好没有结束标志?>,
PHP自身并不需要结束符号,不加结束符让程序更加安全,很大程度防止了末尾被注入额外的内容。
3.5 配置文件
在入口文件中,我们加载了config.php文件的内容,那它有何作用呢?
从名称不难看出,它的作用是保存一些常用配置。
config.php 文件内容如下,作用是定义数据库连接参数参数,以及配置默认控制器名和操作名:
<?php
// 数据库配置
define('DB_NAME', 'project');
define('DB_USER', 'root');
define('DB_PASSWORD', '123456');
define('DB_HOST', 'localhost');
// 默认控制器和操作名
$config['defaultController'] = 'Item';
$config['defaultAction'] = 'index';
return $config;
入口中的$config变量接收到配置参数后,再传给框架的核心类,也就是Fastphp类。
3.6 框架核心类
入口文件对框架类做了两步操作:实例化,调用run()方法。
实例化操作接受$config参数配置,并保存到类属性中。
run()方法则调用用类自身方法,完成:自动加载类文件、监测开发环境、过滤敏感字符、移除全局变量的老用法、和处理路由。
<?php
/**
* fastphp框架核心
*/
class Fastphp
{
protected $_config = [];
public function __construct($config)
{
$this->_config = $config;
}
// 运行程序
public function run()
{
spl_autoload_register(array($this, 'loadClass'));
$this->setReporting();
$this->removeMagicQuotes();
$this->unregisterGlobals();
$this->setDbConfig();
$this->route();
}
// 路由处理
public function route()
{
$controllerName = $this->_config['defaultController'];
$actionName = $this->_config['defaultAction'];
$param = array();
$url = $_SERVER['REQUEST_URI'];
// 清除?之后的内容
$position = strpos($url, &