MVC-Muster (Model-View-Controller) ist ein Software-Architekturmuster in der Softwareentwicklung. MVC unterteilt das Softwaresystem in drei grundlegende Teile: Model (Modell), View (Ansicht) und Controller (Controller). Das MVC-Muster in PHP wird auch „Web MVC“ genannt und entstand in den 1970er Jahren. Der Zweck von MVC besteht darin, ein dynamisches Programmierdesign zu implementieren, das nachträgliche Änderungen, Erweiterungen und Vereinfachungen des Programms erleichtert und die Wiederverwendung bestimmter Programmteile ermöglicht. Darüber hinaus macht dieser Modus die Programmstruktur intuitiver, indem er die Komplexität vereinfacht. Die Funktionen jedes Teils von MVC: Modell
– verwaltet1. Der Controller fängt die vom Benutzer gesendete Anfrage ab;
2 Der Controller ruft das Modell auf, um den Lese- und Schreibvorgang des Status abzuschließen; Der Controller übergibt die Daten an die Ansicht; Ansicht rendert das Endergebnis und präsentiert es dem Benutzer.
2 Warum ein eigenes MVC-Framework entwickeln? Es gibt eine große Anzahl hervorragender MVC-Frameworks im Internet. Dieses Tutorial ist nicht dazu gedacht, eine umfassende und ultimative MVC-Framework-Lösung zu entwickeln. Wir sahen darin eine großartige Gelegenheit, PHP von innen zu lernen. Dabei lernen SieObjektorientierte Programmierung
undMVC-Entwurfsmuster
sowie einige Überlegungen zur Entwicklung kennen. Darüber hinaus kann jeder durch ein selbst erstelltes MVC-Framework sein eigenes Framework vollständig steuern und Ihre Ideen in Ihr Framework integrieren. Ist das nicht eine wunderbare Sache?~~3.1 UmgebungsvorbereitungHier benötigen wir die grundlegendste PHP-Umgebung:
Kleinbuchstaben mit Unterstreichung
sein, wie zum Beispiel:item code>, <code>car_orders
. CarModel
. 3. Controller müssen die CamelCase-Nomenklatur verwenden, also den ersten Buchstaben Großbuchstaben, und Controller
nach dem Namen hinzufügen, z. CarController
. item
,car_orders
。
2、模块名(Models)需用大驼峰命名法,即首字母大写,并在名称后添加Model
,如:ItemModel
,CarModel
。
3、控制器(Controllers)需用大驼峰命名法,即首字母大写,并在名称后添加Controller
,如:ItemController
,CarController
。
4、方法名(Action)需用小驼峰命名法,即首字母小写,如:index
,indexPost
。
5、视图(Views)部署结构为控制器名/行为名,如:item/view.php
,car/buy.php
。
上述规则是为了程序能更好地相互调用。 接下来就开始真正的PHP MVC编程了。
在开始开发前,我们给这个框架先起个名字吧,就叫:Fastphp框架。 然后根据需要来把项目的目录创建。 假设我们建立的项目为 project,目录结构就这样:
project WEB部署根目录 ├─app 应用目录 │ ├─controllers 控制器目录 │ ├─models 模块目录 │ ├─views 视图目录 ├─config 配置文件目录 ├─fastphp 框架核心目录 │ ├─base MVC基类目录 │ ├─db 数据库操作类目录 │ ├─Fastphp.php 内核文件 ├─static 静态文件目录 ├─index.php 入口文件
然后按照下一步,把Nginx或者Apache的站点根目录配置到project
目录,。
重定向的目的有两个:设置根目录为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
4. Der Methodenname (Aktion) muss die
verwenden, d >.
🎜5. Die Bereitstellungsstruktur von Views ist 🎜Controllername/Verhaltensname🎜, wie zum Beispiel:item/view.php
, car/buy.php
. 🎜🎜Die oben genannten Regeln dienen dazu, dass sich Programme besser gegenseitig aufrufen können. Als nächstes beginnt die eigentliche PHP-MVC-Programmierung. 🎜🎜3.3 Verzeichnisvorbereitung🎜🎜🎜Bevor wir mit der Entwicklung beginnen, geben wir diesem Framework einen Namen, nennen wir es: 🎜Fastphp-Framework🎜. Erstellen Sie dann das Projektverzeichnis nach Bedarf. Angenommen, das von uns erstellte Projekt ist 🎜project🎜 und die Verzeichnisstruktur ist wie folgt: 🎜<span class="hljs-title">location</span> / { <span class="hljs-comment"> # 重新向所有非真是存在的请求到index.php </span><span class="hljs-title"> try_files</span> <span class="hljs-variable">$uri</span> <span class="hljs-variable">$uri</span>/ /index.php<span class="hljs-variable">$args</span>; }
project
zu konfigurieren. 🎜🎜3.4 Umleitung🎜🎜🎜Der Zweck der Umleitung ist zweifach: das Stammverzeichnis auf den Speicherort von project
festzulegen und alle Anfragen an die Datei index.php zu senden. Wenn es sich um einen 🎜Apache-Server🎜 handelt, erstellen Sie eine neue 🎜.htaccess🎜-Datei im Projektverzeichnis mit dem Inhalt: 🎜<?php // 应用目录为当前目录 define('APP_PATH', __DIR__ . '/'); // 开启调试模式 define('APP_DEBUG', true); // 加载框架文件 require(APP_PATH . 'fastphp/Fastphp.php'); // 加载配置文件 $config = require(APP_PATH . 'config/config.php'); // 实例化框架类 (new fastphp\Fastphp($config))->run();
server hinzu
Block:🎜<?php // 数据库配置 $config['db']['host'] = 'localhost'; $config['db']['username'] = 'root'; $config['db']['password'] = '123456'; $config['db']['dbname'] = 'project'; // 默认控制器和操作名 $config['defaultController'] = 'Item'; $config['defaultAction'] = 'index'; return $config;
(1)静态文件能直接访问。如果文件或者目录真实存在,则直接访问存在的文件/目录。 比如,静态文件static/css/main.css
真实存在,就可以直接访问它。 (2)程序有单一的入口。 这种情况是请求地址不是真实存在的文件或目录,这样请求就会传到 index.php 上。 例如,访问地址:localhost/item/detail/1
,在文件系统中并不存在这样的文件或目录。 那么,Apache或Nginx服务器会把请求发给index.php
,并且把域名之后的字符串赋值给REQUEST_URI
变量。 这样在PHP中用$_SERVER['REQUEST_URI']
就能拿到/item/detail/1
; (3)可以用来生成美化的URL,利于SEO。
接下来,在 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\Fastphp($config))->run();
注意,上面的PHP代码中,并没有添加PHP结束符号?>
。
这么做的主要原因是:
对于只有 PHP 代码的文件,最好没有结束标志?>
,
PHP自身并不需要结束符号,不加结束符让程序更加安全,很大程度防止了末尾被注入额外的内容。
在入口文件中,我们加载了config.php文件的内容,那它有何作用呢? 从名称不难看出,它的作用是保存一些常用配置。 config.php 文件内容如下,作用是定义数据库连接参数参数,以及配置默认控制器名和操作名:
<?php // 数据库配置 $config['db']['host'] = 'localhost'; $config['db']['username'] = 'root'; $config['db']['password'] = '123456'; $config['db']['dbname'] = 'project'; // 默认控制器和操作名 $config['defaultController'] = 'Item'; $config['defaultAction'] = 'index'; return $config;
入口中的$config
变量接收到配置参数后,再传给框架的核心类,也就是Fastphp
类。
入口文件对框架类做了两步操作:实例化,调用run()方法。
实例化操作接受$config
参数配置,并保存到对象属性中。
run()
方法则调用用类自身方法,完成下面几个操作:
类自动加载
环境检查
过滤敏感字符
移除全局变量的老用法
路由处理
在fastphp
目录下新建核心类文件,名称Fastphp.php,代码:
<?php namespace fastphp; // 框架根目录 defined('CORE_PATH') or define('CORE_PATH', __DIR__); /** * 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, '?'); $url = $position === false ? $url : substr($url, 0, $position); // 删除前后的“/” $url = trim($url, '/'); if ($url) { // 使用“/”分割字符串,并保存在数组中 $urlArray = explode('/', $url); // 删除空的数组元素 $urlArray = array_filter($urlArray); // 获取控制器名 $controllerName = ucfirst($urlArray[0]); // 获取动作名 array_shift($urlArray); $actionName = $urlArray ? $urlArray[0] : $actionName; // 获取URL参数 array_shift($urlArray); $param = $urlArray ? $urlArray : array(); } // 判断控制器和操作是否存在 $controller = 'app\\controllers\\'. $controllerName . 'Controller'; if (!class_exists($controller)) { exit($controller . '控制器不存在'); } if (!method_exists($controller, $actionName)) { exit($actionName . '方法不存在'); } // 如果控制器和操作名存在,则实例化控制器,因为控制器对象里面 // 还会用到控制器名和操作名,所以实例化的时候把他们俩的名称也 // 传进去。结合Controller基类一起看 $dispatch = new $controller($controllerName, $actionName); // $dispatch保存控制器实例化后的对象,我们就可以调用它的方法, // 也可以像方法中传入参数,以下等同于:$dispatch->$actionName($param) call_user_func_array(array($dispatch, $actionName), $param); } // 检测开发环境 public function setReporting() { if (APP_DEBUG === true) { error_reporting(E_ALL); ini_set('display_errors','On'); } else { error_reporting(E_ALL); ini_set('display_errors','Off'); ini_set('log_errors', 'On'); } } // 删除敏感字符 public function stripSlashesDeep($value) { $value = is_array($value) ? array_map(array($this, 'stripSlashesDeep'), $value) : stripslashes($value); return $value; } // 检测敏感字符并删除 public function removeMagicQuotes() { if (get_magic_quotes_gpc()) { $_GET = isset($_GET) ? $this->stripSlashesDeep($_GET ) : ''; $_POST = isset($_POST) ? $this->stripSlashesDeep($_POST ) : ''; $_COOKIE = isset($_COOKIE) ? $this->stripSlashesDeep($_COOKIE) : ''; $_SESSION = isset($_SESSION) ? $this->stripSlashesDeep($_SESSION) : ''; } } // 检测自定义全局变量并移除。因为 register_globals 已经弃用,如果 // 已经弃用的 register_globals 指令被设置为 on,那么局部变量也将 // 在脚本的全局作用域中可用。 例如, $_POST['foo'] 也将以 $foo 的 // 形式存在,这样写是不好的实现,会影响代码中的其他变量。 相关信息, // 参考: http://php.net/manual/zh/faq.using.php#faq.register-globals public function unregisterGlobals() { if (ini_get('register_globals')) { $array = array('_SESSION', '_POST', '_GET', '_COOKIE', '_REQUEST', '_SERVER', '_ENV', '_FILES'); foreach ($array as $value) { foreach ($GLOBALS[$value] as $key => $var) { if ($var === $GLOBALS[$key]) { unset($GLOBALS[$key]); } } } } } // 配置数据库信息 public function setDbConfig() { if ($this->config['db']) { define('DB_HOST', $this->config['db']['host']); define('DB_NAME', $this->config['db']['dbname']); define('DB_USER', $this->config['db']['username']); define('DB_PASS', $this->config['db']['password']); } } // 自动加载类 public function loadClass($className) { $classMap = $this->classMap(); if (isset($classMap[$className])) { // 包含内核文件 $file = $classMap[$className]; } elseif (strpos($className, '\\') !== false) { // 包含应用(application目录)文件 $file = APP_PATH . str_replace('\\', '/', $className) . '.php'; if (!is_file($file)) { return; } } else { return; } include $file; // 这里可以加入判断,如果名为$className的类、接口或者性状不存在,则在调试模式下抛出错误 } // 内核文件命名空间映射关系 protected function classMap() { return [ 'fastphp\base\Controller' => CORE_PATH . '/base/Controller.php', 'fastphp\base\Model' => CORE_PATH . '/base/Model.php', 'fastphp\base\View' => CORE_PATH . '/base/View.php', 'fastphp\db\Db' => CORE_PATH . '/db/Db.php', 'fastphp\db\Sql' => CORE_PATH . '/db/Sql.php', ]; } }
下面重点讲解主请求方法 route()
,它也称路由方法。 路由方法的主要作用是:截取URL,并解析出控制器名、方法名和URL参数。 假设我们的 URL 是这样:
yoursite.com/controllerName/actionName/queryString
当浏览器访问上面的URL,route()
从全局变量 $_SERVER['REQUEST_URI']
中获取到字符串/controllerName/actionName/queryString
。 然后,会将这个字符串分割成三部分:controllerName
、actionName
和 queryString
。 例如,URL链接为:yoursite.com/item/detail/1/hello,那么route()
分割之后,
分割完成后,路由方法再实例化控制器:itemController
,并调用其中的detail
方法 。
接下来,就是在 fastphp 中创建MVC基类,包括控制器、模型和视图三个基类。 在fastphp/base/
目录下新建控制器基类,文件名 Controller.php,功能就是总调度,内容如下:
<?php namespace fastphp\base; /** * 控制器基类 */ class Controller { protected $_controller; protected $_action; protected $_view; // 构造函数,初始化属性,并实例化对应模型 public function __construct($controller, $action) { $this->_controller = $controller; $this->_action = $action; $this->_view = new View($controller, $action); } // 分配变量 public function assign($name, $value) { $this->_view->assign($name, $value); } // 渲染视图 public function render() { $this->_view->render(); } }
Controller
类用assign()
方法实现把变量保存到View对象中。 这样,在调用$this->render()
后视图文件就能显示这些变量。
新建模型基类,继承自数据库操作类Sql
类。 因为数据库操作比较复杂,所以SQL操作我们单独创建一个类。 Model
基类涉及到3个类:Model
基类本身,它的父类SQL
,以及提供数据库连接句柄的Db
类。 在fastphp/base/
目录下新建模型基类文件,名为 Model.php,代码如下:
<?php namespace fastphp\base; use fastphp\db\Sql; class Model extends Sql { protected $model; public function __construct() { // 获取数据库表名 if (!$this->table) { // 获取模型类名称 $this->model = get_class($this); // 删除类名最后的 Model 字符 $this->model = substr($this->model, 0, -5); // 数据库表名与类名一致 $this->table = strtolower($this->model); } } }
在fastphp/db/
目录下建立一个数据库基类 Sql.php,代码如下:
<?php namespace fastphp\db; use \PDOStatement; class Sql { // 数据库表名 protected $table; // 数据库主键 protected $primary = 'id'; // WHERE和ORDER拼装后的条件 private $filter = ''; // Pdo bindParam()绑定的参数集合 private $param = array(); /** * 查询条件拼接,使用方式: * * $this->where(['id = 1','and title=Web', ...])->fetch(); * 为防止注入,建议通过$param方式传入参数: * $this->where(['id = :id'], [':id' => $id])->fetch(); * * @param array $where 条件 * @return $this 当前对象 */ public function where($where = array(), $param = array()) { if ($where) { $this->filter .= ' WHERE '; $this->filter .= implode(' ', $where); $this->param = $param; } return $this; } /** * 拼装排序条件,使用方式: * * $this->order(['id DESC', 'title ASC', ...])->fetch(); * * @param array $order 排序条件 * @return $this */ public function order($order = array()) { if($order) { $this->filter .= ' ORDER BY '; $this->filter .= implode(',', $order); } return $this; } // 查询所有 public function fetchAll() { $sql = sprintf(select * from `%s` %s, $this->table, $this->filter); $sth = Db::pdo()->prepare($sql); $sth = $this->formatParam($sth, $this->param); $sth->execute(); return $sth->fetchAll(); } // 查询一条 public function fetch() { $sql = sprintf(select * from `%s` %s, $this->table, $this->filter); $sth = Db::pdo()->prepare($sql); $sth = $this->formatParam($sth, $this->param); $sth->execute(); return $sth->fetch(); } // 根据条件 (id) 删除 public function delete($id) { $sql = sprintf(delete from `%s` where `%s` = :%s, $this->table, $this->primary, $this->primary); $sth = Db::pdo()->prepare($sql); $sth = $this->formatParam($sth, [$this->primary => $id]); $sth->execute(); return $sth->rowCount(); } // 新增数据 public function add($data) { $sql = sprintf(insert into `%s` %s, $this->table, $this->formatInsert($data)); $sth = Db::pdo()->prepare($sql); $sth = $this->formatParam($sth, $data); $sth = $this->formatParam($sth, $this->param); $sth->execute(); return $sth->rowCount(); } // 修改数据 public function update($data) { $sql = sprintf(update `%s` set %s %s, $this->table, $this->formatUpdate($data), $this->filter); $sth = Db::pdo()->prepare($sql); $sth = $this->formatParam($sth, $data); $sth = $this->formatParam($sth, $this->param); $sth->execute(); return $sth->rowCount(); } /** * 占位符绑定具体的变量值 * @param PDOStatement $sth 要绑定的PDOStatement对象 * @param array $params 参数,有三种类型: * 1)如果SQL语句用问号?占位符,那么$params应该为 * [$a, $b, $c] * 2)如果SQL语句用冒号:占位符,那么$params应该为 * ['a' => $a, 'b' => $b, 'c' => $c] * 或者 * [':a' => $a, ':b' => $b, ':c' => $c] * * @return PDOStatement */ public function formatParam(PDOStatement $sth, $params = array()) { foreach ($params as $param => &$value) { $param = is_int($param) ? $param + 1 : ':' . ltrim($param, ':'); $sth->bindParam($param, $value); } return $sth; } // 将数组转换成插入格式的sql语句 private function formatInsert($data) { $fields = array(); $names = array(); foreach ($data as $key => $value) { $fields[] = sprintf(`%s`, $key); $names[] = sprintf(:%s, $key); } $field = implode(',', $fields); $name = implode(',', $names); return sprintf((%s) values (%s), $field, $name); } // 将数组转换成更新格式的sql语句 private function formatUpdate($data) { $fields = array(); foreach ($data as $key => $value) { $fields[] = sprintf(`%s` = :%s, $key, $key); } return implode(',', $fields); } }
应该说,Sql基类是框架的核心部分。为什么? 因为通过它,我们创建了一个 SQL 抽象层,可以大大减少了数据库的编程工作。 虽然 PDO 接口本来已经很简洁,但是抽象之后框架的可灵活性更高。 Sql
类里面有用到Db:pdo()
方法,这是我们创建的Db
类,它提供一个PDO单例。 在fastphp/db/
目录下创建Db.php文件,内容:
<?php namespace fastphp\db; use PDO; use PDOException; /** * 数据库操作类。 * 其$pdo属性为静态属性,所以在页面执行周期内, * 只要一次赋值,以后的获取还是首次赋值的内容。 * 这里就是PDO对象,这样可以确保运行期间只有一个 * 数据库连接对象,这是一种简单的单例模式 * Class Db */ class Db { private static $pdo = null; public static function pdo() { if (self::$pdo !== null) { return self::$pdo; } try { $dsn = sprintf('mysql:host=%s;dbname=%s;charset=utf8', DB_HOST, DB_NAME); $option = array(PDO::ATTR_DEFAULT_FETCH_MODE => PDO::FETCH_ASSOC); return self::$pdo = new PDO($dsn, DB_USER, DB_PASS, $option); } catch (PDOException $e) { exit($e->getMessage()); } } }
在fastphp/base/
目录下新建视图基类 View.php 内容如下:
<?php namespace fastphp\base; /** * 视图基类 */ class View { protected $variables = array(); protected $_controller; protected $_action; function __construct($controller, $action) { $this->_controller = strtolower($controller); $this->_action = strtolower($action); } // 分配变量 public function assign($name, $value) { $this->variables[$name] = $value; } // 渲染显示 public function render() { extract($this->variables); $defaultHeader = APP_PATH . 'app/views/header.php'; $defaultFooter = APP_PATH . 'app/views/footer.php'; $controllerHeader = APP_PATH . 'app/views/' . $this->_controller . '/header.php'; $controllerFooter = APP_PATH . 'app/views/' . $this->_controller . '/footer.php'; $controllerLayout = APP_PATH . 'app/views/' . $this->_controller . '/' . $this->_action . '.php'; // 页头文件 if (is_file($controllerHeader)) { include ($controllerHeader); } else { include ($defaultHeader); } //判断视图文件是否存在 if (is_file($controllerLayout)) { include ($controllerLayout); } else { echo <h1>无法找到视图文件</h1>; } // 页脚文件 if (is_file($controllerFooter)) { include ($controllerFooter); } else { include ($defaultFooter); } } }
这样,核心的PHP MVC框架核心就完成了。 下面我们编写应用来测试框架功能。
在 SQL 中新建一个 project 数据库,增加一个item
表、并插入两条记录,命令如下:
CREATE DATABASE `project` DEFAULT CHARACTER SET utf8 COLLATE utf8_general_ci; USE `project`; CREATE TABLE `item` ( `id` int(11) NOT NULL auto_increment, `item_name` varchar(255) NOT NULL, PRIMARY KEY (`id`) ) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8; INSERT INTO `item` VALUES(1, 'Hello World.'); INSERT INTO `item` VALUES(2, 'Lets go!');
然后,我们还需要在app/models/
目录中创建一个 ItemModel.php 模型,内容如下:
<?php namespace app\models; use fastphp\base\Model; use fastphp\db\Db; /** * 用户Model */ class ItemModel extends Model { /** * 自定义当前模型操作的数据库表名称, * 如果不指定,默认为类名称的小写字符串, * 这里就是 item 表 * @var string */ protected $table = 'item'; /** * 搜索功能,因为Sql父类里面没有现成的like搜索, * 所以需要自己写SQL语句,对数据库的操作应该都放 * 在Model里面,然后提供给Controller直接调用 * @param $title string 查询的关键词 * @return array 返回的数据 */ public function search($keyword) { $sql = select * from `$this->table` where `item_name` like :keyword; $sth = Db::pdo()->prepare($sql); $sth = $this->formatParam($sth, [':keyword' => %$keyword%]); $sth->execute(); return $sth->fetchAll(); } }
因为 Item
模型继承了 Model
基类,所以它拥有 Model
类的所有功能。
在 app/controllers/
目录下创建一个 ItemController.php 控制器,内容如下:
<?php namespace app\controllers; use fastphp\base\Controller; use app\models\ItemModel; class ItemController extends Controller { // 首页方法,测试框架自定义DB查询 public function index() { $keyword = isset($_GET['keyword']) ? $_GET['keyword'] : ''; if ($keyword) { $items = (new ItemModel())->search($keyword); } else { // 查询所有内容,并按倒序排列输出 // where()方法可不传入参数,或者省略 $items = (new ItemModel)->where()->order(['id DESC'])->fetchAll(); } $this->assign('title', '全部条目'); $this->assign('keyword', $keyword); $this->assign('items', $items); $this->render(); } // 查看单条记录详情 public function detail($id) { // 通过?占位符传入$id参数 $item = (new ItemModel())->where([id = ?], [$id])->fetch(); $this->assign('title', '条目详情'); $this->assign('item', $item); $this->render(); } // 添加记录,测试框架DB记录创建(Create) public function add() { $data['item_name'] = $_POST['value']; $count = (new ItemModel)->add($data); $this->assign('title', '添加成功'); $this->assign('count', $count); $this->render(); } // 操作管理 public function manage($id = 0) { $item = array(); if ($id) { // 通过名称占位符传入参数 $item = (new ItemModel())->where([id = :id], [':id' => $id])->fetch(); } $this->assign('title', '管理条目'); $this->assign('item', $item); $this->render(); } // 更新记录,测试框架DB记录更新(Update) public function update() { $data = array('id' => $_POST['id'], 'item_name' => $_POST['value']); $count = (new ItemModel)->where(['id = :id'], [':id' => $data['id']])->update($data); $this->assign('title', '修改成功'); $this->assign('count', $count); $this->render(); } // 删除记录,测试框架DB记录删除(Delete) public function delete($id = null) { $count = (new ItemModel)->delete($id); $this->assign('title', '删除成功'); $this->assign('count', $count); $this->render(); } }
在 app/views/
目录下新建 header.php 和 footer.php 两个页头页脚模板,如下。 header.php 内容:
<html> <head> <meta http-equiv=Content-Type content=text/html; charset=utf-8 /> <title><?php echo $title ?></title> <link rel=stylesheet href=/static/css/main.css type=text/css /> </head> <body> <h1><?php echo $title ?></h1>
footer.php 内容:
</body> </html>
页头文件用到了main.css样式文件,内容:
html, body { margin: 0; padding: 10px; font-size: 20px; } input { font-family:georgia,times; font-size:24px; line-height:1.2em; } a { color:blue; font-family:georgia,times; line-height:1.2em; text-decoration:none; } a:hover { text-decoration:underline; } h1 { color:#000000; font-size:41px; border-bottom:1px dotted #cccccc; } td {padding: 1px 30px 1px 0;}
然后,在 app/views/item
目录下创建以下几个视图文件。 index.php,浏览数据库内 item
表的所有记录,内容:
<form action= method=get> <input type=text value=<?php echo $keyword ?> name=keyword> <input type=submit value=搜索> </form> <p><a href=/item/manage>新建</a></p> <table> <tr> <th>ID</th> <th>内容</th> <th>操作</th> </tr> <?php foreach ($items as $item): ?> <tr> <td><?php echo $item['id'] ?></td> <td><?php echo $item['item_name'] ?></td> <td> <a href=/item/manage/<?php echo $item['id'] ?>>编辑</a> <a href=/item/delete/<?php echo $item['id'] ?>>删除</a> </td> </tr> <?php endforeach ?> </table>
add.php,添加记录后的提示,内容:
<a class=big href=/item/index>成功添加<?php echo $count ?>条记录,点击返回</a>
manage.php,管理记录,内容:
<form action=<?php echo $postUrl; ?> method=post> <?php if (isset($item['id'])): ?> <input type=hidden name=id value=<?php echo $item['id'] ?>> <?php endif; ?> <input type=text name=value value=<?php echo isset($item['item_name']) ? $item['item_name'] : '' ?>> <input type=submit value=提交> </form> <a class=big href=/item/index>返回</a>
update.php,更改记录后的提示,内容:
<a class=big href=/item/index>成功修改<?php echo $count ?>项,点击返回</a>
delete.php,删除记录后的提示,内容:
<a href=/item/index>成功删除<?php echo $count ?>项,点击返回</a>
这样,在浏览器中访问 project程序:http://localhost/item/index/,就可以看到效果了。
以上代码已经全部发布到 github 上,关键部分加了注释:
Das obige ist der detaillierte Inhalt vonSchreiben eines PHP-MVC-Frameworks [empfohlenes Tutorial für Kindermädchen]. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!