学习代码审计时,接触到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/indexlocalhost/index.php/Hello/name/tom
localhost/index.php/Index/welecome
参考