几个概念:

THINKPHP 是一个MVC框架,使用PATHINFO解析出分组名,模块名,方法名,以及参数。

PATHINFO:就是 http://localhost/index.php/Home/Index/index/a/1/b/2?c=3 中的红色部分,注意,c=3并不是pathinfo的一部分,它是一个query参数。PATHINFO指的是URL中的路径实际上不存在的时候,apache或nginx等通过一定的手段将这个不存在路径保存到环境变量 $_SERVER['PATH_INFO']中,也可以将它转化为$_GET['s']参数,TP也可以识别。
也就是说当你访问THINKPHP项目的时候,你实际上访问的只是入口文件index.php ,不管你后面有多长的路径,都是实际不存在的,转化成PATHINFO或$_GET['s']了而已。

那么THINKPHP就是根据PATHINFO,从中提取出 分组名,模块名,方法名,以及将剩余的部分转化为$_GET变量,比如TP从PATHINFO字符串中提取'Home'字符串,将其保存到$_GET['g'] ,$_GET['m']='Index' ;$_GET['a']='index'  ;$_GET['a']=1  ;$_GET['b']=2。不信你可以随便在某个模块方法下 dump('$_GET')看看。

之后Tp就是根据$_GET['g'],$_GET['m'],$_GET['a']来加载实例化对应分组下的模块类,并执行对应的方法。

关于URL详细的解析流程,请看 ThinkPHP/Lib/Core/Dispatcher.class.php 核心类。

URL路由:

URL路由--Url Router,指的是,在TP进行常规url解析之前,先检测路由,如果发现有路由规则匹配当前的PATHINFO,那么URL解析则交给路由处理。

路由规则是由多条 rule=>router 规则组成的数组。

路由处理过程大概如下:

1.遍历路由规则rule,与当前PATHINFO字符串进行匹配,如果合法,则从PATHINFO中取出所需要的字符串。

2.路由中每条规则对应一个router,router中指定对应的模块以及方法,模块和方法可以用第1步中匹配到的字符串进行动态替代。

3.将剩余的参数都解析并写入到$_GET中。

路由规则:

首先在配置文件中开启路由,"URL_ROUTER_ON"=>true;
然后定义路由规则,路由规则也在配置文件中,参数名为 URL_ROUTE_RULES ,它是一个数组。
假设我们打开一个url:http://localhost/index.php/
list/2/p/1 ,我们需要访问id为1分类列表页,并且p是页码,第一页。
PATHINFO字符串为:
list/2/p/1 
我们需要将它定位到 listAction.class.php 模块中的 index方法,以及将p作为分页参数写入到$_GET中
定义路由:
'URL_ROUTE_RULES'=>array(
'list/:id/p/:p'=>'Home/list/index'

),
这样PATHINFO中的2将被写入到$_GET['id']=2, 同时 p 对应$_GET['p']=1 ,规则重定向到Home分组下的list模块index方法,我们只需要在index方法中直接获取$_GET['id'],以及$_GET['p']即可,实际访问效果等同于  http://localhost/index.php/Home/list/index?id=2&p=1
假如我们要严格限定id和p的类型是数字,那么我们应该这样做
'list/:id
\d/p/:p
\d'=>'Home/list/index'
\d表示数字类型。

复杂一点的,假如PATHINFO为  
del/10 。要调用article模块下的del方法,并将10作为id传递给它
规则应该这样写
“:a/:id\d”=>'Home/article/
:1'
首先,将del这一段作为一个变量去匹配,:a :b都可以,10也作为一个变量,指定为:id 并且类型要求是整型。在右侧router中,通过:1可以使用左侧第一个变量,:2 :3依次类推。
再假设,我们禁止匹配
add/10中的add,规则只需要稍加改动,如下:
“:a^add/:id\d”=>'Home/article/
:1'
如果要禁止多个关键词,则只需要用
|拼接,比如 ^add|del|get 

如果右侧router默认需要带有参数,可以直接将参数加在?后面,比如
“:a^add/:id\d”=>'Home/article/
:1?p=1'

也可以将右侧router写成数组形式,比如
“:a^add/:id\d”=>array('Home/article/:1','p=1&p1=2&p3=3')


如果你需要
完整匹配整个PATHINFO,只需要在rule的结尾加上
$,比如
list/:id$ 可以匹配 list/10,但是无法匹配list/10/p/1
把$拿掉则都可以匹配。


上面所讲的都是我们程序内部重定向,实际URL并不会发生跳转,如果我们想做一个redirect的跳转,只需要右侧router以http://开头或者以 '/' 开头即可。

比如
'list/:id'=>'/artlist/:1' 
这样当我们访问 http://localhost/list/10的时候,浏览器会发生跳转,跳转到 http://localhost/artlist/10

以上是TP自定定义的一些规则,那么当以上规则无法满足你的复杂需求时怎么办?
正则路由可以帮助你 。

正则路由与普通路由的区别是,左侧rule不需要记住复杂的语法,我们只需要根据正则语法,直接匹配PATHINFO,右侧通过 :1 :2 :3调用即可。

比如
'/^list\/(\d+)/is'=>'Home/artlist/index?id=:1'

注意事项:
1.路由不会循环匹配,当匹配一次以后将跳出循环。
2.当路由中同时包含 ^排除关键字 和 \指定类型的时候,应当将 ^放在\前面,比如 :d^add|del\s  




ThinkPHP中URL解析原理,以及URL路由使用教程!的更多相关文章

  1. ThinkPHP中的pathinfo模式和URL重写

    语文一直不太好,要我怎么解释这个pathinfo模式还真不知道怎么说,那就先来一段代码说下pathinfo模式吧 http://serverName/appName/module/action/id/ ...

  2. Thinkphp中路由Url获取的使用方法

    Thinkphp是一个体系较为完整的框架,很多地方比国外的框架功能都全,唯一不喜之处是性能,和传说中的.NET有点像. Thinkphp提供较全url处理体系,通过同一规则实现Url的路由和Url生成 ...

  3. 在ASP.NET非MVC环境中(WebForm中)构造MVC的URL参数,以及如何根据URL解析出匹配到MVC路由的Controller和Action

    目前项目中有个需求,需要在WebForm中去构造MVC的URL信息,这里写了一个帮助类可以在ASP.NET非MVC环境中(WebForm中)构造MVC的URL信息,主要就是借助当前Http上下文去构造 ...

  4. ThinkPHP中pathinfo模式与URL重写

    Thinkphp中的pathinfo模式 http://serverName/appName/module/action/id/1/ pathinfo模式 在不考虑路由的情况下,第一个参数会被解析成模 ...

  5. thinkphp中URL传参数的几种方式

    在thinkphp中,url传参合asp.net中原理类似,下面就单个参数和多个参数传递方式进行一个简单讲解 1.传单个参数 单个参数这种比较简单,例如 想像edit操作里面传递一个id值,如下写法_ ...

  6. java网络编程客户端与服务端原理以及用URL解析HTTP协议

    常见客户端与服务端 客户端: 浏览器:IE 服务端: 服务器:web服务器(Tomcat),存储服务器,数据库服务器. (注:会用到Tomact服务器,在webapps下有一个自己创建的目录myweb ...

  7. 域名解析中的cname解析和显性URL跳转和隐性URL跳转三者有什么区别

    通俗的来讲,cname解析还是属于dns解析,只是把某个域名解析到另外一个域名对应的某个IP的空间中,所以还需要在服务器端(比如nginx)做域名解析(比如把baidu.com做一个cname解析到i ...

  8. ThinkPHP中U方法与url的四种访问模式

     ThinkPHP中U方法的用处主要是完成对url地址的组装,在模板中使用U方法而不是固定写死URL地址的好处在于,一旦你的环境变化或者参数设置改变,你不需要更改模板中的任何代码.在模板中的调用格式需 ...

  9. 「Python 编程」编码实现网络请求库中的 URL 解析器

    摘要:怎么写出更短的代码并不是这次要讨论的话题.今天我们来研究一下:运行代码的计算机是如何找到目标服务器的? 相信各位 Python 开发者都用过 Requests 库,有些朋友还用过 WebSock ...

随机推荐

  1. C++重载输入和输出操作符以及IO标准库中的刷新输入缓冲区残留字符问题

    今天在做C++ Primer习题的14.11时,印象中应该挺简单的一题,结果却费了很长时间. 类定义: typedef string Date; class CheckoutRecord{ publi ...

  2. FPGA中latency与delay概念的区别

    2013-06-17 21:09:26 最近学习流水线以及状态机,总遇到注入加入寄存器可以分割组合逻辑,从而提高电路的运行频率的说法:还有流水线可以提高速度的说法,刚开始很是疑惑,觉得流水线的方法,输 ...

  3. NFC(8)关于新买的标签的格式化

    有多种方法格式化nfc标签设备. 如搜相关的手机上应用,在应用里选择格式类型 本文是用代码手动格式 public void writeNFCTag(Tag tag) { if (tag == null ...

  4. createSQLQuery与createQuery的区别

    本文原址 : http://stta04.javaeye.com/blog/377633hibernate 中createQuery与createSQLQuery 昨晚帮同事看代码到凌晨2点多,今早6 ...

  5. Android 多种方式正确的加载图像,有效避免oom

    图像加载的方式: Android开发中消耗内存较多一般都是在图像上面,本文就主要介绍怎样正确的展现图像减少对内存的开销,有效的避免oom现象.首先我们知道我的获取图像的来源一般有三种源头:1.从网络加 ...

  6. 百度UEditor编辑器使用教程与使用方法

    我们在做网站的时候,网站后台系统一般都会用到web编辑器,今天笔者就给大家推荐一款百度UEditor编辑器.关于这款百度UEditor编辑器官网上也有简单的教程,不过看着比较费劲,今天笔者就跟大家分享 ...

  7. 数学(扩展欧几里得算法):HDU 5114 Collision

    Matt is playing a naive computer game with his deeply loved pure girl. The playground is a rectangle ...

  8. 八、jdk工具之JvisualVM、JvisualVM之一--(visualVM介绍及性能分析示例)

    目录 一.jdk工具之jps(JVM Process Status Tools)命令使用 二.jdk命令之javah命令(C Header and Stub File Generator) 三.jdk ...

  9. 【译】 AWK教程指南 9读取命令行上的参数

    大部分的应用程序都允许使用者在命令之后增加一些选择性的参数.执行awk时这些参数大部分用于指定数据文件文件名,有时希望在程序中能从命令行上得到一些其它用途的数据.本小节中将叙述如何在awk程序中取用这 ...

  10. 线性表-串:KMP模式匹配算法

    一.简单模式匹配算法(略,逐字符比较即可) 二.KMP模式匹配算法 next数组:j为字符序号,从1开始. (1)当j=1时,next=0: (2)当存在前缀=后缀情况,next=相同字符数+1: ( ...