-
首先需要说明的是,URL写作
/index.php/xx/b/c
可以直接被index.php
文件接受,而不用做任何额外的操作 -
但是如果需要在URL中隐藏
index.php
的话,才需要伪静态重写规则,将^(.*)$
重写到/index.php/$1
-
首先,需要从URI拆分键值对(此处省略)
-
下面的思路是(依次):
-
获取到要调用函数的参数名
-
按照参数名称排序URI拆分出的键值对
-
使用
call_user_func_array()
方法,动态调用函数
-
-
由于键值对经过了按照参数名顺序排序,所以新的URI键值对中,值的顺序恰好是和需要调用函数的顺序对应的,所以可以直接用
call_user_func_array()
调用 -
下面,介绍一下其中需要用到的知识:
-
call_user_func_array
-
array_fill_keys
-
ReflectionFunction或者ReflectionMethod
-
由于Reflection部分缺少官方文档,贴出Overflow的回答作为示例:
-
-
下面上代码:
function test(....) {}...$funcRef = new ReflectionFunction('test');# 或者$funcRef = new ReflectionMethod('class', 'test');$paramNames = [];foreach($funcRef->getParameters() as $param) { $paramNames[] = $param->name;}$params = array_fill_keys($paramNames, NULL);# 使用NULL为默认值,$paramNames为键名,填充一个参数键值对# 如果参数对应项不传值,则那个值默认是NULL(暂不考虑有默认值的情况)# 假设从URI获取的键值对生成的数组为$URIParam# 将$URIParam中的值填入上面生成的参数数组中# 遍历$params在$URIParam中找的原因是,如果遍历$URIParam,对应的$params中的值都是NULL,无法通过值确定键名是否存在# 而遍历$params的话,键名一定是参数列表中的,只需要看在$URIParams中是否存在即可foreach($params as $paramName => $param) { if(isset($URIParam[$paramName])) { # 使用isset是为了避免直接判断而产生的等效假值问题 # 如果用if($URIParam[$paramName])无法区分0和''这种等效于假值的值 $params[$paramName] = $param; # 由于,$params中的值已经定义过了,所以此处赋值并不会影响其顺序 }}call_user_func('test', $params);# 或者call_user_func_array([$class, 'method'], $params);