当前位置:Gxlcms > PHP教程 > php之mvc框架演进过程详解_PHP教程

php之mvc框架演进过程详解_PHP教程

时间:2021-07-01 10:21:17 帮助过:6人阅读

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简易框架就到这里差不多了。


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

\

\



www.bkjia.comtruehttp://www.bkjia.com/PHPjc/477363.htmlTechArticle1) /********** php与html混编完成增删改查功能**************************/ 1.设计思路: 根据平时练习一个增删改查的功能进行,即在一个php文件中完...

人气教程排行