自制PHP框架之路由与控制器


Posted in PHP onMay 07, 2017

我们为什么要使用路由?原因1:一个更漂亮的URI

1.URI的改进

刚刚开始学PHP时,我们一定写过blog.php?id=1之类的URI,使用GET方式获取参数。这样的URI有两个缺点,一是容易被SQL注射攻击,二是维护性可读性差,大家可以比较下面两种URI哪一种更具备可读性。

www.mysite.com/blog.php?id=1

上面URI是我们初学PHP最常用的。

www.mysite.com/blog/1

这种URI是目前最流行的URI,举个例子,比如很多读书类,电影类网站,都使用了这样的URI,这样的URI要比index.php?a=1&b=2&c=3&d=4....要简洁很多。

2.实现方法

在WEB项目的根目录下写一个.htaccess文件

RewriteEngine On

RewriteRule ^([a-zA-Z0-9/]*)$ index.php/$1

重写规则,让域名后面的字符串直接做为一个参数传入index.php,这样index.php就成为了你整个WEB应用的中心,定义了“请求和响应的映射”。

原因2:单一入口机制的易维护性

1.路由数组

一个PHP初学者,刚开始做项目,项目做着做着规模做大了,常常这个PHP页面给另一个PHP页面用GET方法传值,有时传的值还不止一个,时间一久,你的WEB项目,N个PHP页面宛如一个复杂的蜘蛛网,让你难以维护。一旦有修改,会涉及很多PHP文件,工作量很大。

MVC的单一入口机制可以解决维护难的问题,路由就是一套映射,可以让你一个URI对应一个方法。

$route=[

  ''=>'IndexController@Index',

  'blog'=>'BlogController@Show',

  'blog/{id}/{name}'=>'BlogController@Show',

];

2.获取参数

$path=$_SERVER['PATH_INFO'];

$path=ltrim($path,'/');

echo $path.PHP_EOL;

我们在浏览器里输入:www.mysite.com/blog/1后,path变量为/blog/1。使用ltrim函数删除左边的斜杠,然后使用explode把字符串拆解成数组。

$path_arr=explode('/', $path);

核心代码如下:

if(isset($_SERVER['PATH_INFO'])){

  $path=$_SERVER['PATH_INFO'];

  $path=ltrim($path,'/');

  $path_arr=explode('/', $path);

}

 

if(isset($path_arr[0])){

  $key=$path_arr[0];

  unset($path_arr[0]);

}

else{

  $key='';

}

 

if(isset($path_arr[1])){

  $parameters=array_values($path_arr);

}

 

 

if(isset($route[$key])){

  $arr=explode('@', $route[$key]);

   

  $controller=new $arr[0];

  $action=$arr[1];

   

  if(isset($parameters)){

    $controller->$action($parameters);

  }

  else{

    $controller->$action();

  }  

}

else{

  require 'error.html.php';

}

unset函数可以销毁数组中key和value,但是并不会重建索引,所以path_arr[0]是要调用的控制器类和方法名,path_arr[1]或者path_arr[1..N]就作为传入方法的参数。

重定向和错误页面是WEB系统中最常见的,如果不用路由机制,你可能要没完没了的重复写重定向或者错误页面的显示或者跳转代码,有了路由,只需要一句话就可以完成。

原因3:减少资源的消耗

MVC采用了控制器(controller)来响应请求(request),每次请求来时,应该在指定的一个PHP文件中初始化这个控制器,而不是分别在不同的PHP文件中做初始化工作,这样可以减少资源的消耗。

是不是一定要用控制器?方案1:不用控制器

我们现在路由数组里添加一项,value不是一个字符串,而是一个匿名函数(Closure)

$route=[

  ''=>'Index',

  'blog'=>'BlogController@Show',

  'blog/{id}/{name}'=>'BlogController@Show',

  'f'=>function(){echo 'hello';}

]; 

这里的route[f]是一个匿名函数,并不是一个控制器类的方法,所以,我们要把上一节路由代码做一下修改:

if(isset($route[$key])){

  if($route[$key] instanceof Closure){

    $route[$key]();

  }

  else{

    $arr=explode('@', $route[$key]);  

    $controller=new $arr[0];

    $action=$arr[1];  

    if(isset($parameters)){

      $controller->$action($parameters);

    }

    else{

      $controller->$action();

    }

  }

}

else{

  require 'error.html.php';

}

方案2:使用控制器

自制PHP框架之路由与控制器

每一次都require一个html页面是一件很不优雅的事情,所以我们写一个render函数

function render($path,array $args){

  extract($args);

  require($path);

}

接上一篇博客,我们知道每个URI对应了一个方法,但是我们常常遇到这样的问题:

<?php 

 

class Controller{

  public function __call($method,$args){

    echo 'has not this function'.$method;

  }

}

 

class IndexController extends Controller{

  public function Index(){

    echo __CLASS__;

    for($i=1;$i<=20;++$i){

      $data[$i]='content';

    }

 

    render('template.html.php',['data'=>$data]);

  }

}

 

class BlogController extends Controller{

  public function Show(){

    echo __CLASS__;

    for($i=1;$i<=10;++$i){

      $data[$i]='blog';

    }

    render('template.html.php',['data'=>$data]);

  }

}

 

?>

用不用控制器,取决于你的业务复杂度。个人建议使用控制器,但是对于业务很简单的页面跳转或检查,可以直接写在一个匿名函数里。

控制器里写些什么?

我们也许写过这样的代码:

class IndexController extends Controller{

  public function Index($content){

    return '<html><head></head><body>'.$content.'</body></html>';

  }

}

这样把界面的代码嵌入的写法是非常难以维护的,也是很多开发人员(包括我)最厌恶的写法,因为这种写法并没有做好界面与业务逻辑的分离,所以我们需要使用视图。

<html>

  <head>

   

  </head>

   

  <body>

    <?php foreach($data as $key=>$value){ ?>  

      <div>

        <?php echo $key.':'.$value; ?>  

      </div>

    <?php } ?>

  </body>

</html>

每一次调用控制器的某个方法时,render函数都会把参数以关联数组的形式传入,做到“业务逻辑”和“表现”的浅层次分离,但是这种分离还不是最好的,因为前端开发人员仍然需要面对甚至处理PHP代码,后端开发人员也有和前端人员沟通的成本,所以后面某一节,会再谈一种更好的分离方式。

PHP 相关文章推荐
PHP用户指南-cookies部分
Oct 09 PHP
php下实现折线图效果的代码
Apr 28 PHP
PHP URL地址获取函数代码(端口等) 推荐
May 15 PHP
PHP中获取文件扩展名的N种方法小结
Feb 27 PHP
header跳转和include包含问题详解
Sep 08 PHP
PHP弹出提示框并跳转到新页面即重定向到新页面
Jan 24 PHP
PHP的preg_match匹配字符串长度问题解决方法
May 03 PHP
PHP实现的简单三角形、矩形周长面积计算器分享
Nov 18 PHP
Zend Framework使用Zend_Loader组件动态加载文件和类用法详解
Dec 09 PHP
php+Memcached实现简单留言板功能示例
Feb 15 PHP
PHP实现的折半查询算法示例
Oct 09 PHP
Laravel框架在本地虚拟机快速安装的方法详解
Jun 11 PHP
PHP-CGI远程代码执行漏洞分析与防范
May 07 #PHP
PHP关键特性之命名空间实例详解
May 06 #PHP
PHP 中使用explode()函数切割字符串为数组的示例
May 06 #PHP
Thinkphp 空操作、空控制器、命名空间(详解)
May 05 #PHP
thinkPHP实现的联动菜单功能详解
May 05 #PHP
thinkPHP实现的省市区三级联动功能示例
May 05 #PHP
php str_replace替换指定次数的方法详解
May 05 #PHP
You might like
PHP错误和异长常处理总结
2014/03/06 PHP
PHP的RSA加密解密方法以及开发接口使用
2018/02/11 PHP
Laravel基础-关于引入公共文件的两种方式
2019/10/18 PHP
浅析PHP反序列化中过滤函数使用不当导致的对象注入问题
2020/02/15 PHP
javascript之解决IE下不渲染的bug
2007/06/29 Javascript
XML的代替者----JSON
2007/07/21 Javascript
基于JQuery的浮动DIV显示提示信息并自动隐藏
2011/02/11 Javascript
ajax不执行success回调而是执行了error回调
2012/12/10 Javascript
用js实现放大镜的效果的简单实例
2016/05/23 Javascript
JavaScript动态检验密码强度的实现方法
2016/11/09 Javascript
浅析script标签中的defer与async属性
2016/11/30 Javascript
浅谈React Native 中组件的生命周期
2017/09/08 Javascript
使用webpack-dev-server处理跨域请求的方法
2018/04/18 Javascript
vue中实现上传文件给后台实例详解
2019/08/22 Javascript
ionic+html5+API实现双击返回键退出应用
2019/09/17 Javascript
linux 下以二进制的方式安装 nodejs
2020/02/12 NodeJs
echarts 使用formatter 修改鼠标悬浮事件信息操作
2020/07/20 Javascript
Vue实现购物小球抛物线的方法实例
2020/11/22 Vue.js
[03:46]显微镜下的DOTA2第七期——满血与残血
2014/06/20 DOTA
在Python的struct模块中进行数据格式转换的方法
2015/06/17 Python
在Django中编写模版节点及注册标签的方法
2015/07/20 Python
Python使用Selenium+BeautifulSoup爬取淘宝搜索页
2018/02/24 Python
Django自定义过滤器定义与用法示例
2018/03/22 Python
wxPython的安装与使用教程
2018/08/31 Python
Python面向对象之继承和多态用法分析
2019/06/08 Python
浅谈django框架集成swagger以及自定义参数问题
2020/07/07 Python
Python中猜拳游戏与猜筛子游戏的实现方法
2020/09/04 Python
Python监听剪切板实现方法代码实例
2020/11/11 Python
html2canvas把div保存图片高清图的方法示例
2018/03/05 HTML / CSS
英国Amara家居法国网站:家居装饰,现代装饰和豪华礼品
2016/12/15 全球购物
经济信息管理专业大学生求职信
2013/09/27 职场文书
和谐社区口号
2014/06/19 职场文书
对外汉语专业大学生职业生涯规划范文
2014/09/13 职场文书
三峡大坝导游词
2015/01/31 职场文书
生产车间主任岗位职责
2015/04/08 职场文书
少先队工作总结2015
2015/05/13 职场文书