学习代码审计时,接触到mvc框架的路由功能模式,简单记录一下。
MVC路由机制
按照传统,在很多Web框架中(如经典的ASP、JSP、PHP、ASP.NET等之类的框架),URL代表的是磁盘上的物理文件。
例如,当看到请求http://example.com/albums/list.aspx时,我们可以确定该站点目录结构中含有一个albums文件夹,并且在该文件夹下还有一个list.aspx文件。 URL和文件系统之间这种一一对应的关系并不适用于大部分基于MVC的Web框架,如ASP.NET MVC.一般来说,这些框架采用不同的方法将URL映射到某个类上的方法调用,而不是映射到磁盘上的物理文件。 另外,对于MVC应用程序,URL 请求到达的第一个组件是控制器而不是视图,而控制器是没有物理路径的。
路由机制概述
1.匹配传入的请求(该请求不匹配服务器文件系统中文件),并将这些请求映射到控制器操作(Controller中的action方法)
MVC基本的处理流程:来了一个URL请求, 从中找到Controller和Action的值, 将请求传递给Controller处理. Controller获取Model数据对象, 并且将Model传递给View, 最后View负责呈现页面。(说白了,就是来了一个URL,找到一个控制器中的方法)(路由是模式,有参数,通过URL中的参数,就可以对应找到符合这种路由模式的方法)
Routing的作用:
´URL: localhost/home/index
´localhost是域名, 所以首先要去掉域名部分: home/index
´对应了上面代码中的这种URL结构: {controller}/{action}/{id}
´因为我们建立了这种URL结构的识别规则, 所以能够识别出 Controller是home, action是index, id没有则为默认值"".
路由机制一般是由路由名称,路由模式和默认值构成的
Defaults属性
最一般的路由情况
routes.MapRoute(
name: "Default",
url: "{controller}/{action}/{id}",
defaults: new { controller = "Home", action = "GlobalIndex", id = UrlParameter.Optional }
);
MVC 模拟路由机制搭建
1、首先,建立好MVC文件结构
下面我们就来动手试试,怎么才能访问到controllers里面的文件。
在index.php里面输入以下内容:
<?php /** * MVC路由功能简单实现 * @desc 简单实现MVC路由功能 * $Author: Zhihua_W */
//打印出所有的服务器变量
print_r($_SERVER);
?>
然后输入访问地址试试。这里我是用本地环境的,我访问的地址是:localhost/MVC/index.php/class/function/var1。下面我贴出两个最重要的变量:
[REQUEST_URI] => /MVC/index.php/class/function/var1
[SCRIPT_NAME] => /MVC/index.php
其实路由最基本的原理就在这里,通过这2个变量来提取url地址里的class名称和 function名称,参数等,然后把class include进来,通过PHP的回调函数 call_user_func_array 调用对应的function和传递相应的参数。接下来就是各个文件的具体代码了,读代码应该比我写的易懂。
2、index.php文件内容
<?php /** * MVC路由功能简单实现 * @desc 简单实现MVC路由功能 * $Author: Zhihua_W */ //定义application路径 define('APPPATH', trim(__DIR__ . '/')); //获得请求地址 $root = $_SERVER['SCRIPT_NAME']; $request = $_SERVER['REQUEST_URI']; $URI = array(); //获得index.php 后面的地址 $url = trim(str_replace($root, '', $request), '/'); //如果为空,则是访问根地址 if (empty($url)) { //默认控制器和默认方法 $class = 'index'; $func = 'welcome'; } else { $URI = explode('/', $url); //如果function为空 则默认访问index if (count($URI) < 2) { $class = $URI[0]; $func = 'index'; } else { $class = $URI[0]; $func = $URI[1]; } } //把class加载进来 include(APPPATH . '/' . 'application/controllers /' . $class . '.php'); //实例化->将控制器首字母大写 $obj = ucfirst($class); call_user_func_array( //调用内部function array($obj, $func), //传递参数 array_slice($URI, 2) ); ?>
3、在application/controllers里面添加下面2个文件(index.php,hello.php)
index.php
<?php /** * MVC路由功能简单实现 * @desc 简单实现MVC路由功能 * $Author: Zhihua_W */ class Index { function welcome() { # code... echo "default covtroller!"; } }
hello.php
<?php /** * MVC路由功能简单实现 * @desc 简单实现MVC路由功能 * $Author: Zhihua_W */ class Hello{ public function index(){ echo "hello world!"; } public function name($name){ echo "hello ".$name; } } ?>
4、测试
分别访问localhost/index.php/Hello/index
localhost/index.php/Hello/name/tom
localhost/index.php/Index/welecome
参考