[TOC]
* * * * *
## 1 Url操作文件源代码(thinkphp\library\think\Url.php)
~~~
public static function build($url = '', $vars = '', $suffix = true, $domain = false)
{
$info = parse_url($url);
$url = !empty($info['path']) ? $info['path'] : '';
if (isset($info['fragment'])) {
$anchor = $info['fragment'];
if (false !== strpos($anchor, '?')) {
list($anchor, $info['query']) = explode('?', $anchor, 2);
}
if (false !== strpos($anchor, '@')) {
list($anchor, $domain) = explode('@', $anchor, 2);
}
} elseif (false !== strpos($url, '@')) {
list($url, $domain) = explode('@', $info['path'], 2);
}
if (is_string($vars)) {
parse_str($vars, $vars);
}
if (isset($info['query'])) {
parse_str($info['query'], $params);
$vars = array_merge($params, $vars);
}
$alias = self::getRouteAlias();
if (0 !== strpos($url, '/') && isset($alias[$url]) && $match = self::getRouteUrl($alias[$url], $vars)) {
$url = str_replace('[--think--]', '', $match);
} else {
$url = self::parseUrl($url);
}
$type = Route::bind('type');
if ($type) {
$bind = Route::bind($type);
if (0 === strpos($url, $bind)) {
$url = substr($url, strlen($bind) + 1);
}
}
$depr = Config::get('pathinfo_depr');
$url = str_replace('/', $depr, $url);
$suffix = ('/' == $url) ? '' : self::parseSuffix($suffix);
$anchor = !empty($anchor) ? '#' . $anchor : '';
if (!empty($vars)) {
if (Config::get('url_common_param')) {
$vars = urldecode(http_build_query($vars));
$url .= $suffix . $anchor . '?' . $vars;
} else {
foreach ($vars as $var => $val) {
if ('' !== trim($val)) {
$url .= $depr . $var . $depr . urlencode($val);
}
}
$url .= $suffix . $anchor;
}
} else {
$url .= $suffix . $anchor;
}
$domain = self::parseDomain($url, $domain);
$url = $domain . Config::get('base_url') . '/' . $url;
return $url;
}
~~~
~~~
protected static function parseUrl($url)
{
if (0 === strpos($url, '/')) {
$url = substr($url, 1);
} elseif (false !== strpos($url, '\\')) {
$url = ltrim(str_replace('\\', '/', $url), '/');
} elseif (0 === strpos($url, '@')) {
$url = substr($url, 1);
} else {
$module = MODULE_NAME ? MODULE_NAME . '/' : '';
if ('' == $url) {
$url = $module . CONTROLLER_NAME . '/' . ACTION_NAME;
} else {
$path = explode('/', $url);
$len = count($path);
if ($len < 3) {
$url = $module . (1 == $len ? CONTROLLER_NAME . '/' : '') . $url;
}
}
}
return $url;
}
~~~
~~~
protected static function parseDomain($url, $domain)
{
if ($domain) {
if (true === $domain) {
$domain = $_SERVER['HTTP_HOST'];
if (Config::get('url_domain_deploy')) {
$domain = $_SERVER['HTTP_HOST'];
foreach (Route::domain() as $key => $rule) {
$rule = is_array($rule) ? $rule[0] : $rule;
if (false === strpos($key, '*') && 0 === strpos($url, $rule)) {
$domain = $key . strstr($domain, '.');
break;
}
}
}
} else {
$domain .= strpos($domain, '.') ? '' : strstr($_SERVER['HTTP_HOST'], '.');
}
$domain = (self::isSsl() ? 'https://' : 'http://') . $domain;
} else {
$domain = '';
}
return $domain;
}
~~~
~~~
protected static function pattern($pattern, $vars)
{
foreach ($pattern as $key => $type) {
if (1 == $type && !isset($vars[$key])) {
return false;
}
}
return true;
}
~~~
~~~
protected static function parseSuffix($suffix)
{
if ($suffix) {
$suffix = true === $suffix ? Config::get('url_html_suffix') : $suffix;
if ($pos = strpos($suffix, '|')) {
$suffix = substr($suffix, 0, $pos);
}
}
return (empty($suffix) || 0 === strpos($suffix, '.')) ? $suffix : '.' . $suffix;
}
~~~
~~~
public static function isSsl()
{
if (isset($_SERVER['HTTPS']) && ('1' == $_SERVER['HTTPS'] || 'on' == strtolower($_SERVER['HTTPS']))) {
return true;
} elseif (isset($_SERVER['SERVER_PORT']) && ('443' == $_SERVER['SERVER_PORT'])) {
return true;
}
return false;
}
~~~
~~~
public static function getRouteUrl($alias, &$vars = [])
{
foreach ($alias as $key => $val) {
list($url, $pattern) = $val;
if (strpos($url, '$')) {
$url = str_replace('$', '[--think--]', $url);
}
if (self::pattern($pattern, $vars)) {
foreach ($vars as $key => $val) {
if (false !== strpos($url, '[:' . $key . ']')) {
$url = str_replace('[:' . $key . ']', $val, $url);
unset($vars[$key]);
} elseif (false !== strpos($url, ':' . $key)) {
$url = str_replace(':' . $key, $val, $url);
unset($vars[$key]);
}
}
return $url;
}
}
return false;
}
~~~
~~~
private static function getRouteAlias()
{
if ($alias = Cache::get('think_route_alias')) {
return $alias;
}
$rules = Route::getRules();
if (empty($rules)) {
return [];
}
foreach ($rules as $rule => $val) {
if (!empty($val['routes'])) {
foreach ($val['routes'] as $key => $route) {
if (is_numeric($key)) {
$key = array_shift($route);
}
$route = $route[0];
if (is_array($route)) {
$route = implode('\\', $route);
} elseif ($route instanceof \Closure) {
continue;
} elseif (strpos($route, '?')) {
$route = strstr($route, '?', true);
}
$var = self::parseVar($rule . '/' . $key);
$alias[$route][] = [$rule . '/' . $key, $var];
}
} else {
$route = $val['route'];
if (is_array($route)) {
$route = implode('\\', $route);
} elseif ($route instanceof \Closure) {
continue;
} elseif (strpos($route, '?')) {
$route = strstr($route, '?', true);
}
$var = self::parseVar($rule);
$alias[$route][] = [$rule, $var];
}
}
Cache::set('think_route_alias', $alias);
return $alias;
}
~~~
~~~
private static function parseVar($rule)
{
if ($depr = Config::get('url_params_depr')) {
$rule = str_replace($depr, '/', $rule);
}
$var = [];
foreach (explode('/', $rule) as $val) {
$optional = false;
if (0 === strpos($val, '[:')) {
$optional = true;
$val = substr($val, 1, -1);
}
if (0 === strpos($val, ':')) {
$name = substr($val, 1);
$type = $optional ? 2 : 1;
$var[$name] = $type;
}
}
return $var;
}
~~~
~~~
public static function clearAliasCache()
{
Cache::rm('think_route_alias');
}
~~~
## 2 文件分析
1 `public static function build($url = '', $vars = '', $suffix = true, $domain = false){}`
生成Url
> $url:url基础表达式
> $vars:url参数
> $suffix:伪静态后缀
> $domain:域名参数
2 `protected static function parseUrl($url){}`
解析url地址
> $url:待解析url参数
3 `protected static function parseDomain($url, $domain){}`
解析url地址,域名
> $url:待解析url参数
> $domain:待解析domain域名参数
4 `protected static function pattern($pattern, $vars){}`
解析路由规则中的变量是否有传入
> $pattern: 路由规则
> $vars: 规则参数
5 `protected static function parseSuffix($suffix){}`
解析url后缀
> $suffix:url后缀参数
6 `public static function isSsl(){}`
判断是否开启Ssl
7 `public static function getRouteUrl($alias, &$vars = []){}`
匹配路由地址
> $alias: 路由别名
> $vars: 路由规则参数
8 `private static function getRouteAlias(){}`
生成路由别名并缓存
9 `private static function parseVar($rule){}`
分析路由规则中的变量
10 `public static function clearAliasCache(){}`
清空路由别名缓存
## 3 使用方法
TODO:
## 4 总 结
4个public方法
> build()
> isSsl()
> getRouteUrl()
> clearAliasCache
4个protected方法
> parseUrl()
> parseDomain()
> pattern()
> parseSuffix()
2个private方法
> getRouteAlias()
> parseVar()
- 更新记录
- 概述
- 文件索引
- 函数索引
- 章节格式
- 框架流程
- 前:章节说明
- 主:(index.php)入口
- 主:(start.php)框架引导
- 主:(App.php)应用启动
- 主:(App.php)应用调度
- C:(Controller.php)应用控制器
- M:(Model.php)数据模型
- V:(View.php)视图对象
- 附:(App.php)应用启动
- 附:(base.php)全局变量
- 附:(common.php)模式配置
- 附:(convention.php)全局配置
- 附:(Loader.php)自动加载器
- 附:(Build.php)自动生成
- 附:(Hook.php)监听回调
- 附:(Route.php)全局路由
- 附:(Response.php)数据输出
- 附:(Log.php)日志记录
- 附:(Exception.php)异常处理
- 框架工具
- 另:(helper.php)辅助函数
- 另:(Cache.php)数据缓存
- 另:(Cookie.php)cookie操作
- 另:(Console.php)控制台
- 另:(Debug.php)开发调试
- 另:(Error.php)错误处理
- 另:(Url.php)Url操作文件
- 另:(Loader.php)加载器实例化
- 另:(Input.php)数据输入
- 另:(Lang.php)语言包管理
- 另:(ORM.php)ORM基类
- 另:(Process.php)进程管理
- 另:(Session.php)session操作
- 另:(Template.php)模板解析
- 框架驱动
- D:(\config)配置解析
- D:(\controller)控制器扩展
- D:(\model)模型扩展
- D:(\db)数据库驱动
- D:(\view)模板解析
- D:(\template)模板标签库
- D:(\session)session驱动
- D:(\cache)缓存驱动
- D:(\console)控制台
- D:(\process)进程扩展
- T:(\traits)Trait目录
- D:(\exception)异常实现
- D:(\log)日志驱动
- 使用范例
- 服务器与框架的安装
- 控制器操作
- 数据模型操作
- 视图渲染控制
- MVC开发初探
- 模块开发
- 入口文件定义全局变量
- 运行模式开发
- 框架配置
- 自动生成应用
- 事件与插件注册
- 路由规则注册
- 输出控制
- 多种应用组织
- 综合应用
- tp框架整合后台auto架构快速开发
- 基础原理
- php默认全局变量
- php的魔术方法
- php命名空间
- php的自动加载
- php的composer
- php的反射
- php的trait机制
- php设计模式
- php的系统时区
- php的异常错误
- php的输出控制
- php的正则表达式
- php的闭包函数
- php的会话控制
- php的接口
- php的PDO
- php的字符串操作
- php的curl
- 框架心得
- 心:整体结构
- 心:配置详解
- 心:加载器详解
- 心:输入输出详解
- 心:url路由详解
- 心:模板详解
- 心:模型详解
- 心:日志详解
- 心:缓存详解
- 心:控制台详解
- 框架更新
- 4.20(验证类,助手函数)
- 4.27(新模型Model功能)
- 5.4(新数据库驱动)
- 7.28(自动加载)