直接通过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. 《LoadRunner12七天速成宝典》签售会2016-12-17北京

    报名地址: http://www.after615.com/actives/s?id=3141&time=1480042829608&sign=9ac8e25e9ab3cf57f613 ...

  2. ASP.Net MVC——使用 ITextSharp 完美解决HTML转PDF(中文也可以)

    前言: 最近在做老师交代的一个在线写实验报告的小项目中,有这么个需求:把学生提交的实验报告(HTML形式)直接转成PDF,方便下载和打印. 以前都是直接用rdlc报表实现的,可这次牵扯到图片,并且更为 ...

  3. Maven常用命令

    开发中常用的命令: 1. mvn compile 编译源代码2. mvn test-compile 编译测试代码3. mvn test 运行测试4. mvn package 打包,根据pom.xml打 ...

  4. linux字符串url编码与解码

    编码的两种方式 echo '手机' | tr -d '\n' | xxd -plain | sed 's/\(..\)/%\1/g' echo '手机' |tr -d '\n' |od -An -tx ...

  5. Windows平台Go调用DLL的坑

    最近的项目中,使用了GO来开发一些服务中转程序.业务比较简单,但是有一些业务需要复用原有C++开发的代码.而在WINDOWS,用CGO方式来集成C/C++代码并不是太方便.所以用DLL把C++的代码封 ...

  6. 玩转Windows服务系列——给Windows服务添加COM接口

    当我们运行一个Windows服务的时候,一般情况下,我们会选择以非窗口或者非控制台的方式运行,这样,它就只是一个后台程序,没有界面供我们进行交互. 那么当我们想与Windows服务进行实时交互的时候, ...

  7. ASP.NET MVC Model绑定(三)

    ASP.NET MVC Model绑定(三) 前言 看过前两篇的朋友想必对Model绑定有个大概的了解,然而MVC框架给我们提供了更高的可扩展性的提供程序编程模式,也就是本篇的主题了,会讲解一下Mod ...

  8. WCF学习之旅—实现REST服务(二十二)

    一.什么是REST 表述性状态转移(Representational State Transfer,REST),不是一种标准,而是一种软件架构风格. 基于REST的服务与基于SOAP的服务相比,性能. ...

  9. HTML5_05之SVG扩展、地理定位、拖放

    1.SVG绘图总结: ①方法一:已有svg文件,<img src="x.svg">  方法二:<body><svg></svg>&l ...

  10. 基于Caffe的Large Margin Softmax Loss的实现(上)

    小喵的唠叨话:在写完上一次的博客之后,已经过去了2个月的时间,小喵在此期间,做了大量的实验工作,最终在使用的DeepID2的方法之后,取得了很不错的结果.这次呢,主要讲述一个比较新的论文中的方法,L- ...