直接通过url参数访问业务模块($app)中控制器($ctl)里的函数($act)

我们支持3种路由模式

普通模式

_a=$app,     _u=$ctl.$act

最简单的方式,专注实现业务$act函数,不需要再写额外代码

为什么参数名前面要加下划线就不解释了

easy模式

_easy=$app.$tpl.$ctl.$act

_easy=$app.$ctl.$act

在web开发中,通常我们在$act中输出一个前端页面,

easy模式下,如果未实现$act处理函数,会自动寻找并显示对应的前端模板文件。

对于简单的展示页面适合使用这种路由模式

   url重写模式(需要nginx或apache配置) 

apache: ^rewrite[\.\/](.*)$ /index.php?_rewrite=$1  [R,QSA]

nginx:  rewrite ^/rewrite[\.\/](.*)$ /index.php?_rewrite=$1 last;

rewrite.{$app}.{$ctl}.{$act}.{$params}.html

或更加优雅的目录式访问方式

rewrite/{$app}/{$ctl}/{$act}/{$params}.html

其中$params为选填参数部分.格式为urlencode后的参数列表

如果想传递sp_uid=1&d=1.2&p=sb, 那么$params = sp_uid%3D1%26d%3D1.2%26p%3Ds%2Fb

或sp_uid/1/d/1.2/p/sb

在某些要求url中不能带?&特殊字符的场景下可以使用这种模式

1. 为了能通过qq oauth2登陆验证,需要配置重写规则

rewrite.thirdlogin.index.qqcallback.sp_uid%3D1.php

2. 资源静态化

rewrite.upload.index.out.uidm%3D310ef4b.png

3. 支付回调

rewrite.pay.weixin.native2_notify.php

4. 微信开放平台授权回调

rewrite/web/component/message/_app_id/xxxxxxx.php

部分实现代码

1
2
3
4
5
6
7
8
9
10
11
12
13
    $a = (!empty($_REQUEST['_a']) && is_string($_REQUEST['_a'])) ? $_REQUEST['_a'] : 'web';
    if (!preg_match('/^[\w\.]+$/'$a)) {
        exit('invalid _app name! ' . htmlspecialchars($a));
    }   
    $GLOBALS['_UCT']['APP'] = !empty($a) ? strtolower($a) : 'web';
        
    $u = (!empty($_REQUEST['_u']) && is_string($_REQUEST['_u'])) ? $_REQUEST['_u'] : 'index.index';
    if (!preg_match('/^[\w\.]+$/'$u)) {
        exit('invalid _url name! ' . htmlspecialchars($u));
    }   
    $u                      explode('.'$u, 2); 
    $GLOBALS['_UCT']['CTL'] = !empty($u['0']) ? strtolower($u['0']) : 'index';
    $GLOBALS['_UCT']['ACT'] = !empty($u['1']) ? strtolower($u['1']) : 'index';

easy模式

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
    //easy 模式直接访问模板tpl
    if (!empty($_REQUEST['_easy']) && is_string($_REQUEST['_easy'])) {
        $easy explode('.'$_REQUEST['_easy']);
        switch (count($easy)) {
            case 4:
                $_GET['_u'] = $_REQUEST['_u'] = $easy[2] . '.' $easy[3];
                if (preg_match('/^[\w\.]+$/'$easy[1])) {
                    $GLOBALS['_UCT']['TPL'] = $easy[1];
                }
                $_GET['_a'] = $_REQUEST['_a'] = $easy[0];
                break;
            case 3:
                $_GET['_u'] = $_REQUEST['_u'] = $easy[1] . '.' $easy[2];
                $_GET['_a'] = $_REQUEST['_a'] = $easy[0];
                break;
            case 2:
                $_GET['_u'] = $_REQUEST['_u'] = $easy[1];
                $_GET['_a'] = $_REQUEST['_a'] = $easy[0];
                break;
            case 1:
                $_GET['_a'] = $_REQUEST['_a'] = $easy[0];
                break;
            default:
                exit('invalid _easy param! ' . htmlspecialchars($_REQUEST['_easy']));
        }
    }

rewrite模式

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
    //url重写模式
    if (!empty($_REQUEST['_rewrite']) && is_string($_REQUEST['_rewrite'])) {
        //1.支持apache 重写模式下?后的参数缺失的情况 
        if(stripos($_SERVER['SERVER_SOFTWARE'], 'nginx') === false) {
            $_REQUEST['_rewrite'] = urldecode(substr($_SERVER['QUERY_STRING'], strlen('_rewrite=')));
        }
        //2. 丢弃_rewrite中的后缀名
        $rewrite substr($_REQUEST['_rewrite'], 0, strrpos($_REQUEST['_rewrite'], '.'));
 
        //3. 支持/作为分隔符
        $sp '.';
        for($i = 0; $i strlen($rewrite); $i++) {
            if(in_array($rewrite[$i], array('.''/'))) {
                $sp $rewrite[$i];
                break;
            }
        }
        $rewrite explode($sp$rewrite, 4);
        //最后1段是必填后缀名
        switch(count($rewrite)) {
            case 3:
            case 4: {
                $_GET['_a'] = $_REQUEST['_a'] = $rewrite[0];
                $_GET['_u'] = $_REQUEST['_u'] = $rewrite[1].'.'.$rewrite[2];
                if(!empty($rewrite[3])) {
                    if(strpos($rewrite[3], '/')) {
                        $params explode('/'$rewrite[3]);
                        for($i=0; $i+1<count($params); $i+=2) {
                            $_REQUEST[urldecode($params[$i])] = urldecode($params[$i+1]);
                        }   
                    }   
                    else {
                        foreach(explode('&'$rewrite[3]) as $p) {
                            list($k$v) = explode('='$p, 2);  
                            $_REQUEST[urldecode($k)] = urldecode($v);
                        }   
                    }   
                }   
                break;
            }   
            case 2:
                $_GET['_a'] = $_REQUEST['_a'] = $rewrite[0];
                $_GET['_u'] = $_REQUEST['_u'] = $rewrite[1];
                break;
            case 1: 
                $_GET['_a'] = $_REQUEST['_a'] = $rewrite[0];
                break;
            default:
                break;
        }     
    }

如何实现一个php框架系列文章【4】url路由管理的更多相关文章

  1. 如何实现一个php框架系列文章【开篇】

    1.本系列文章的目的 实现一个小而美的产品级别php框架 自己动手实现一个新框架仅用于学习交流,不打算替代市面上现有的其他主流框架. 2. 我要一个怎样的PHP框架 简单实用,安全优雅,博采众长 安装 ...

  2. 如何实现一个php框架系列文章【5】安全处理输入

    所有的外部输入参数都应该检查合法性. 未正确处理输入数据将可能导致sql注入等漏洞. 框架提供系列函数来取$_REQUEST中的值 requestInt requestString requestFl ...

  3. 如何实现一个php框架系列文章【1】如何组织文件

    1. 组织文件布局应该考虑什么问题 前后端分离 各业务模块分离但方便互相调用 上传文件安全性 方便接入第三方类库 如图所示 框架入口文件 uct/index.php 框架目录         uct/ ...

  4. 如何实现一个php框架系列文章【6】mysql数据库

    实现一个mysql数据库封装需要考虑的问题 使用方便性 采用直接sql语句操作方式.只要会写sql语句,那么将没有其他学习成本. uctphp框架提供的dba辅助封装类,用会之后将爱不释手. 使用前需 ...

  5. 如何实现一个php框架系列文章【2】实现类的自动加载

    根据前一篇文章的设计原则,我们暂时把php文件分为3类,类名和文件名都遵守如下约定.   类名 文件名 路径 模型类m {$app}Mod  {$app}.mod.php {$app}/model   ...

  6. 如何实现一个php框架系列文章【3】支持psr4的自动加载类

    psr4自动加载规范https://github.com/PizzaLiu/PHP-FIG/blob/master/PSR-4-autoloader-cn.md 我们把第三方使用psr规范的类库放在v ...

  7. 2019 年起如何开始学习 ABP 框架系列文章-开篇有益

    2019 年起如何开始学习 ABP 框架系列文章-开篇有益 [[TOC]] 本系列文章推荐阅读地址为:52ABP 开发文档 https://www.52abp.com/Wiki/52abp/lates ...

  8. Artech的MVC4框架学习——第二章URL路由

    总结:HttpModule 和HttpHandler是Asp.net管道的两个重要组件.请求最终处理通过HttpHandler完成.MVC就是通过名为MvcHandler自定义HttpHandler现 ...

  9. 【开源】EFW框架系列文章索引

    开源轻量级.Net框架EnterpriseFrameWork详解 ——自己动手写框架 ——适合中小企业的开发框架 ——Ajax+JqueryEasyUI+NotNetBar+MVC+WebServic ...

随机推荐

  1. Lind.DDD.Aspects通过Plugins实现方法的动态拦截~Lind里的AOP

    回到目录 .Net MVC之所以发展的如些之好,一个很重要原因就是它公开了一组AOP的过滤器,即使用这些过滤器可以方便的拦截controller里的action,并注入我们自己的代码逻辑,向全局的异常 ...

  2. Block解析(iOS)

    1. 操作系统中的栈和堆 我们先来看看一个由C/C++/OBJC编译的程序占用内存分布的结构: 栈区(stack):由系统自动分配,一般存放函数参数值.局部变量的值等.由编译器自动创建与释放.其操作方 ...

  3. 浅谈SQL注入风险 - 一个Login拿下Server

    前两天,带着学生们学习了简单的ASP.NET MVC,通过ADO.NET方式连接数据库,实现增删改查. 可能有一部分学生提前预习过,在我写登录SQL的时候,他们鄙视我说:“老师你这SQL有注入,随便都 ...

  4. grunt配置任务

    这个指南解释了如何使用 Gruntfile 来为你的项目配置task.如果你还不知道 Gruntfile 是什么,请先阅读 快速入门 指南并看看这个Gruntfile 实例. Grunt配置 Grun ...

  5. 杂项之python描述符协议

    杂项之python描述符协议 本节内容 由来 描述符协议概念 类的静态方法及类方法实现原理 类作为装饰器使用 1. 由来 闲来无事去看了看django中的内置分页方法,发现里面用到了类作为装饰器来使用 ...

  6. 我的MYSQL学习心得(八) 插入 更新 删除

    我的MYSQL学习心得(八) 插入 更新 删除 我的MYSQL学习心得(一) 简单语法 我的MYSQL学习心得(二) 数据类型宽度 我的MYSQL学习心得(三) 查看字段长度 我的MYSQL学习心得( ...

  7. ASP.NET Aries JSAPI 文档说明:AR.Utility

    AR.Utility 文档 1:方法: 名称 说明 queryString function (key) *模拟.NET的Request对象 stringFormat function (str, a ...

  8. C#执行异步操作的几种方式比较和总结

    C#执行异步操作的几种方式比较和总结 0x00 引言 之前写程序的时候在遇到一些比较花时间的操作例如HTTP请求时,总是会new一个Thread处理.对XxxxxAsync()之类的方法也没去了解过, ...

  9. 【腾讯Bugly干货分享】React Native项目实战总结

    本文来自于腾讯bugly开发者社区,非经作者同意,请勿转载,原文地址:http://dev.qq.com/topic/577e16a7640ad7b4682c64a7 “8小时内拼工作,8小时外拼成长 ...

  10. (转)利用libcurl获取新浪股票接口, ubuntu和openwrt实验成功(三)

    1.  利用 CURLOPT_WRITEFUNCTION 设置回调函数, 利用 CURLOPT_WRITEDATA 获取数据指针 官网文档如下 CALLBACK OPTIONS CURLOPT_WRI ...