欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  php教程

php之mvc框架演进过程详解

程序员文章站 2022-03-26 13:16:58
...
1) /********** php与html混编完成增删改查功能**************************/

1.设计思路:

根据平时练习一个增删改查的功能进行,即在一个php文件中完成,对数据库的连接操作

及在php文件中展示html代码。html提交到当前页面的php部分进行处理入库动作。

$_SERVER['SCRIPT_FILENAME']包含当前脚本的路径。

这在页面需要指向自己时非常有用。

区别于__FILE__常量包含当前脚本(例如包含文件)的完整路径和文件名。

第一个类:增加商品文件:addProduct.php

if(isset($_POST['submit'])){

$name= $_POST['name'];

$price= $_POST['price'];

$description= $_POST['description'];

//连接数据库操作

mysql_connect('localhost','root','master');

mysql_select_db('test');

mysql_query('set names gbk');

$query= "insert into product values(null,'$name','$price','$description')";

mysql_query($query);

mysql_close();

}

?>

'>

产品名称

产品价格

产品描述

思路:都是在同一文件(当前文件中)操作数据库和展示:

Mysql_fetch_assoc 返回数组,一般需要放到一个空数组内组成一个二维数组返回到页面 mysql_num_rows返回行数

查询数据一般常用:

Php中

$result = Mysql_query(“select * fromproduct order by id desc”);

$data = array();

While ($row = Mysql_fecth_assoc($result)){

$data[]= $row;

}

Html中:

……

查询商品文件: listProduct.php

//连接数据库操作

mysql_connect('localhost','root','master');

mysql_select_db('test');

mysql_query('set names gbk');

$query = "select * from product order by id desc";

$result = mysql_query($query);

$data = array();

while($row = mysql_fetch_assoc($result)){

$data[]= $row;

}

?>

编号

名称

价格

描述

删除

更新

foreach($dataas$row):

?>

">删除

">更新

思路:接下来就进行删除和更新操作,同样是请求到新的php一个文件中来处理删除和更新操作。新建php文件:delProduct.php

delProduct.php:

//连接数据库操作

mysql_connect('localhost','root','master');

mysql_select_db('test');

mysql_query('set names gbk');

$id = $_GET['id'];

$query = "delete from product where id = '$id'";

mysql_query($query);

mysql_close();

//同时跳到listProduct展示结果

header('location:listProduct.php');

更新操作:

updateProduct.php(以下类只是作为展示,提交修改还需要一个php文件,为了不再增加一个文件,修改提到本页面,在action中增加一个参数区别展示和提交修改的操作)

//连接数据库操作

mysql_connect('localhost','root','master');

mysql_select_db('test');

mysql_query('set names gbk');

$id = $_GET['id'];

$query = "select * from product where id = '$id'";

$result = mysql_query($query);

$row = mysql_fetch_row($result);

?>

"/>

产品名称"/>

产品价格"/>

产品描述"/>

修后类内容如下:

//连接数据库操作

mysql_connect('localhost','root','master');

mysql_select_db('test');

mysql_query('set names gbk');

if(isset($_REQUEST['flag'])){

$id= $_POST['id'];

$name= $_POST['name'];

$price= $_POST['price'];

$description= $_POST['description'];

$query= "update product set name = '$name', price = '$price' , description = '$description' where id='$id'";

mysql_query($query);

mysql_close();

header('location:listProduct.php');

}else{

$id= $_GET['id'];

$query= "select * from product where id = '$id'";

$result= mysql_query($query);

$row= mysql_fetch_row($result);

}

?>

"/>

产品名称"/>

产品价格"/>

产品描述"/>

/**************加入DB类封装对数据库的操作***************/

此时,已经完成了商品的增删改查基本功能了,但是,有很多的冗余代码,比如数据库的操作,这时可以考虑把数据的操作提取出来:

DB.class.php:

/**

* 数据库操作类

* @author heyongjia

*

*/

class DB{

private$host ='localhost';

private$username ='root';

private$password ='master';

private$dbname ='test';

private$setCoding ='set names gbk';//注意编码的设置,有时候会出现插入不成功的情况

private$conn;//连接资源

private$result;//结果集

public functionconnect(){

$this->conn =mysql_connect($this->host,$this->username,$this->password);

mysql_select_db($this->dbname);

mysql_query($this->setCoding);

}

public functionquery($query){

$this->result = mysql_query($query);

}

public functionclose(){

mysql_close();

}

}

然后分别改造addProduct.php、delProduct.php、listProduct.php、updateProduct.php操作数据库部分,把每个文件中连接数据,查询的部分提取到数DB.class.php中,修改之后如下:

addProduct.php

if(isset($_POST['submit'])){

$name= $_POST['name'];

$price= $_POST['price'];

$description= $_POST['description'];

$query= "insert into product values(null,'$name','$price','$description')";

$db =newDB();

$db->connect();

$db->query($query);

$db->close();

}

delProduct.php:

include 'class/DB.class.php';

$id = $_GET['id'];

$query = "delete from product where id = '$id'";

$db = newDB();

$db->connect();

$db->query($query);

$db->close();

//同时跳到listProduct展示结果

header('location:listProduct.php');

listProduct.php:

//连接数据库操作

include 'class/DB.class.php';

$query = "select * from product order by id desc";

$db = newDB();

$db->connect();

$db->query($query);

$data = array();

while($row = $db->fetch_assoc()){

$data[]= $row;

}

$db->close();

?>

/***********引用product对象进行数据传递,即数据模型*********************/

思路:增删改查的操作属于业务逻辑,应该封装起来。

封装到类Product.class.php

class Product {

private$id;

private$name;

private$price;

private$description;

public function__set($name,$value){

$this->$name= $value;

}

public function__get($name){

return$this->$name;

}

public functionadd(){

$query= "insert into product values(null,'$this->name','$this->price','$this->description')";

$db= newDB();

$db->connect();

$db->query($query);

$db->close();

}

}

然后在addProduct.php中引入该类,并修改操作数据库动作:

$product= newProduct();

$product->name = $name;

$product->price = $price;

$product->description = $description;

$product->add();

其它的操作数据库方法类似。

思路:通过观察addProduct.php、delProduct.php、listProduct.php、updateProduct.php都有类似的代码,那么可不可以整合在一起呢,完成所有模块的相关请求。

新建一个product.php文件,addProduct.php、delProduct.php中的相关操作数据库的代码拷贝到product.php文件中,并且在product.php中以action参数来区分不同的请求。

//用于完成所有相关product操作的请求

include 'class/DB.class.php';

include 'class/Product.class.php';

$action = $_GET['action'];

if($action =='add'){

//增加

$name= $_POST['name'];

$price= $_POST['price'];

$description= $_POST['description'];

$product= newProduct();

$product->name = $name;

$product->price = $price;

$product->description = $description;

$product->add();

}

if($action =='del'){

//删除

$id= $_GET['id'];

$product= newProduct();

$product->id = $id;

$product->del();

//同时跳到listProduct展示结果

header('location:listProduct.php');

}

这时可以把addProduct.php中的php代码部分删除了,并把html部分独立出来一个addProduct.html文件,

同时也可以把delProduct.php文件删除,因为这两个php文件的php代码都移动了product.php中了。

/********************加入Smarty模拟类库******************************/

先把以上两个功能先完成,

1. 访问addProduct.html文件,完成增加操作,并跳转回addProduct.html页面。访问listProduct.php文件,查看增加的结果。

2. 访问listProduct.php文件,并修改删除链接的请求指向。并同时完成删除的功能。

此时product.php的代码如下:

//用于完成所有相关product操作的请求

include 'class/DB.class.php';

include 'class/Product.class.php';

$action = $_GET['action'];

if($action =='add'){

//增加

$name= $_POST['name'];

$price= $_POST['price'];

$description= $_POST['description'];

$product= newProduct();

$product->name = $name;

$product->price = $price;

$product->description = $description;

$product->add();

header('location:addProduct.html');

}

if($action =='del'){

//删除

$id= $_GET['id'];

$product= newProduct();

$product->id = $id;

$product->del();

//同时跳到listProduct展示结果

header('location:listProduct.php');

}

/**************加入smarty类库************************/

思路:此时已经把增加和删除功能已经集合到了product.php中了,还有查询和更新操作,由于listProduct.php中包含了显示数据列表的功能,不容易展示(可以采用include一个页来完成。)。此时,可以用smarty来完成,这样就合理了。

这时可以考虑把addProduct.html也放入product.php来展示,修改成:

include 'class/Template.class.php';//注意导入该文件,该文件是模拟smarty实一类文件。(考虑到查询功能也用要到,需要展示多个数据,所以模拟Smarty的类文件无法胜任,就直接引入smarty来作为展示)

/***********************加入smarty类库********************/

这时可以把自己的写的Templates类继承Smarty类,这样就可以使用Smarty中的功能,还可以扩展其功能。

Template.class.php内容如下:

/**

* 继承了Smarty类的Template类

* @author heyongjia

*

*/

class TemplateextendsSmarty{

/**

* 调用父类的构造方法,确保父类中的构造方法已经执行

* 因为子类如果有构造方法,就不会执行父类中的构造方法

*/

function__construct(){

parent::__construct();

//初始化模板目录

$this->setTemplateDir('tpl');

}

}

至此,CRUD功能已经整合在一product.php文件中了,也说是所有的请求都经过它,如:

Localhost/test/mvc/product.php?action=add/list/update等请求。这时product.php就相当于一个控制器的角色了。

/**************** 加入入口文件 *******************************、

我们程序中,会有很多的控制器,为了便于管理,我们定义一个入口文件,也就是说,所有的请求都经过这个入口文件,再由这个入口进行分发到各个控制器。

增一个index.php文件:

用于分发到控制器:

请求的url:

Localhost/test/mvc/index.php?module=product&action=add/list/update

所以index.php文件如下:

$module = $_GET['module'];

$action = $_GET['action'];

$file = $module.'Control.php?action='.$action;

header('location:'.$file);

此时,就可以通过上面的入口文件分发到不同的控制器(模块)上的不同的操作(功能),但是由于使用的是header跳转,在浏览器上地址栏就变成了跳转后的地址了,即脱离了入口文件。

那么,如何实现?

将product.php改成一个类来实现。改造product.php为ProductControl.class.php,并移动control文件夹下,内容如下:

class ProductControl{

public functionaddok(){

//增加

$name= $_POST['name'];

$price= $_POST['price'];

$description= $_POST['description'];

$product= newProduct();

$product->name = $name;

$product->price = $price;

$product->description = $description;

$product->add();

header('location:product.php?action=list');

}

public functionadd(){

$smarty= newTemplate();

//$tpl = new Template();

//$tpl->display('addProduct.html');

$smarty->display('addProduct.html');

}

public functiondel(){

//删除

$id= $_GET['id'];

$product= newProduct();

$product->id = $id;

$product->del();

//同时跳到listProduct展示结果

header('location:product.php?action=list');

}

public functionupdate(){

$smarty= newTemplate();

$id= $_GET['id'];

$product= newProduct();

$product->id = $id;

$row= $product->getRow();

$smarty->assign('id',$row['id']);

$smarty->assign('name',$row['name']);

$smarty->assign('price',$row['price']);

$smarty->assign('description',$row['description']);

$smarty->display('updateProduct.html');

}

public functionupdateok(){

$id= $_POST['id'];

$name= $_POST['name'];

$price= $_POST['price'];

$description= $_POST['description'];

$product= newProduct();

$product->id = $id;

$product->name = $name;

$product->price = $price;

$product->description = $description;

$product->update();

header('location:product.php?action=list');

}

}

此时,如何调用这个类的方法呢?在index.php中实例化该对象调用。

//用于完成所有相关product操作的请求

include 'class/DB.class.php';

include 'class/Product.class.php';

include 'smarty/Smarty.class.php';

include 'class/Template.class.php';

$module = $_GET['module'];

$module = ucfirst($module);

$action = $_GET['action'];

$className = $module.'Control';

include 'control/'.$className.'.class.php';

$class = new$className; //根据模块名引用类文件并实例化该对象

$class->$action(); //调用对应的action方法。

修改调用的所有路径为:index.php?module=product&action=xxx

addProduct.html updateProduct.htmllistProduct.html

优化:一般地不会在index.php文件中写太多的代码,可以将代码移动配置文件中,在index.php中引入即可。

一般地,整个工程会有一个主控制器类,用于处理分发控制器,Application.class.php:

class Application{

static functionrun(){

global$module; //函数内要使用外部的变量

global$action;

$className= $module.'Control';

include'control/'.$className.'.class.php';

$class= new$className;

$class->$action();

}

}

init.php:

//用于完成所有相关product操作的请求

include 'class/DB.class.php';

include 'class/Product.class.php';

include 'smarty/Smarty.class.php';

include 'class/Template.class.php';

include 'control/Application.class.php';

$module = $_GET['module'];

$module = ucfirst($module);

$action = $_GET['action'];

Index.php:

include 'config/init.php';

Application::run();

此时,访问的url为: index.php?module=product&action=xxx

但如果直接访问index.php就会报错:


Notice: Undefined index: module in F:\amp\apache\studydocs\test\php_js\mvc\config\init.phpon line10

Notice: Undefined index: action in F:\amp\apache\studydocs\test\php_js\mvc\config\init.phpon line12

这是因为在init.php中没有对module和action进行初始化。

并将值配置到config文件夹下的conig.php文件中。

由于在product.php文件中(其实就是模型类),有很多类似的操作数据库的代码,所以可以将这部分代码封装到父类实现,该模型类只需要继承即可。

/******************封装核心文件core**************************

新建一个文件夹core,将DB.class.php和Application.php移动到core因为是共公的文件,不属于功能文件。

新建一个基类文件(模型基类),用于实现对数据库实例化的封装,而子类模型只需要继承即可获取数据库对象的实例。

Model.class.php:

class Medel{

protected$db;

public function__construct(){

$this->db =newDB();

}

}

新建文件夹model,然后将product.php移动到model文件夹下,并改名ProductModel.class.php:

class ProductModelextendsModel {

private$id;

private$name;

private$price;

private$description;

public function__set($name,$value){

$this->$name= $value;

}

public function__get($name){

return$this->$name;

}

public functionadd(){

$query= "insert into product values(null,'$this->name','$this->price','$this->description')";

$this->db->connect();

$this->db->query($query);

$this->db->close();

}

public functiondel(){

$query= "delete from product where id = '$this->id'";

$this->db->connect();

$this->db->query($query);

$this->db->close();

}

public functionselect(){

$query= "select * from product order by iddesc";

$this->db->connect();

$this->db->query($query);

$data= array();

while($row = $this->db->fetch_assoc()){

$data[]= $row;

}

$this->db->close();

return$data;

}

public functionupdate(){

$query= "update product set name = '$this->name', price = '$this->price' , description = '$this->description' where id='$this->id'";

$this->db->connect();

$this->db->query($query);

$this->db->close();

}

public functiongetRow(){

$query= "select * from product where id = '$this->id'";

$this->db->connect();

$this->db->query($query);

$row= $this->db->fetch_assoc();

$this->db->close();

return$row;

}

}

/***********************视图封装**************************

在控制器中每次都要实例化Template类,并调用display方法。

把Template.class.php移动core文件夹中,并改名为View.class.php:

这么做的好处,让每一个控制继承于该类(构造一个控制器基类Control.class.php,在类中构造一个视图实例),只要每个控制器继承了该类(Control.class.php),即拥有了smarty实例。


/**

* 1.继承了Smarty类的Template类

* 2.不用继承Smarty了,是smarty的一个实例

* @author heyongjia

*

*/

class Viewextends Smarty{

}

这么做的好处,让每一个控制继承于该类(构造一个控制器基类Control.class.php,在类中构造一个视图实例),只要每个控制器继承了该类(Control.class.php),即拥有了smarty实例。

/**

* 控制器基类

* @author heyongjia

*

*/

class Controlextends View{

protected$view;

public function__construct(){

$this->view = new View();

$this->view->setTemplateDir('tpl');

}

}


优化,直接在init.php中直接包含模型是不对的,因为可以出现很多的模型。这时可以考虑自动加载机载来实现。注意:smarty3.0中也用到了自动加载函数,所以要利用注册来完成。

function autoload1($classname){

$filename= 'model/'.$classname.'.class.php';

if(is_file($filename)){

include"$filename";

}

}

spl_autoload_register('autoload1');

function addslashes_func(&$str){

$str= addslashes($str);

}

if(!get_magic_quotes_gpc()){

array_walk_recursive($_POST,'addslashes_func');

array_walk_recursive($_GET,'addslashes_func');

}

此至,就mvc简易框架就到这里差不多了。


用图总结一下,这样就比较直观了: