namespace think;use think\Config;use think\Loader;use think\Request;use think\Route;// 使用 think 里面,内置的 Config Loader Request Routeclass Url{    // 生成URL地址的root    protected static $root;// 根地址    /**     * URL生成 支持路由反射     * @param string            $url 路由地址     * @param string|array      $vars 参数(支持数组和字符串)a=val&b=val2... ['a'=>'val1', 'b'=>'val2']     * @param string|bool       $suffix 伪静态后缀,默认为true表示获取配置值     * @param boolean|string    $domain 是否显示域名 或者直接传入域名     * @return string     */    public static function build($url = '', $vars = '', $suffix = true, $domain = false)    {// url 支持路由反射 $url 路由地址 $vars 参数 a=val b=val2  后缀 是否显示域名        if (false === $domain && Config::get('url_domain_deploy')) {// 如果配置文件支持 域名 设置的话,可以设置过来            $domain = true;        }        // 解析URL        if (0 === strpos($url, '[') && $pos = strpos($url, ']')) {// 如果是用 路由规则的方式 [name]            // [name] 表示使用路由命名标识生成URL            $name = substr($url, 1, $pos - 1);// 获取 路由标识 字符串            $url  = 'name' . substr($url, $pos + 1); // 拼接 成为 新的 字符串        }        $info = parse_url($url);// 根据 url 解析,暂时中断一下        //var_dump(parse_url('home/index/index'));        //array(1) { ["path"]=> string(16) "home/index/index" }        // 中断一下  系统函数  应该会解析返回一个数组 关系        // 尝试一下 parse_url('home/index/index')---->输出什么        $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 (strpos($url, '@')) {// 存在域名            // 解析域名            list($url, $domain) = explode('@', $url, 2);        }        // 解析参数        if (is_string($vars)) {            // aaa=1&bbb=2 转换成数组            parse_str($vars, $vars);// 解析字符串 为数组 默认推荐传入数组格式        }        if ($url) {// 如果 url 存在            $rule = Route::name(isset($name) ? $name : $url . (isset($info['query']) ? '?' . $info['query'] : ''));            // 规则 等于 路由优先 有路由先看路由,然后用 url 拼接参数            if (is_null($rule) && isset($info['query'])) {// 如果 url 为空,但是设置了参数                $rule = Route::name($url);// 获取 name                // 解析地址里面参数 合并到vars                parse_str($info['query'], $params);// 解析参数                $vars = array_merge($params, $vars);// 合并参数                unset($info['query']);//删除信息            }        }        if (!empty($rule) && $match = self::getRuleUrl($rule, $vars)) {// 如果规则不为空,并且 也能匹配到对应的路由标识            // 匹配路由命名标识            $url = $match[0];// 分别 返回 url            if (!empty($match[1])) {                $domain = $match[1];//域名            }        } elseif (!empty($rule) && isset($name)) {// 如果不为空 设置名字            throw new \InvalidArgumentException('route name not exists:' . $name);// 抛出异常        } else {            if (isset($info['query'])) {                // 解析地址里面参数 合并到vars                parse_str($info['query'], $params);                $vars = array_merge($params, $vars);            }            // 路由标识不存在 直接解析            $url = self::parseUrl($url, $domain);// 路由不存在,就直接解析        }        // 检测URL绑定        $type = Route::getBind('type');        if ($type) {            $bind = Route::getBind($type);            if (0 === strpos($url, $bind)) {                $url = substr($url, strlen($bind) + 1);            }        }        // 还原URL分隔符        $depr = Config::get('pathinfo_depr');        $url  = str_replace('/', $depr, $url);        // URL后缀        $suffix = in_array($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 . '?' . $vars . $anchor;            } 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组装        $url = $domain . (self::$root ?: Request::instance()->root()) . '/' . ltrim($url, '/');        return $url;        // 此处的重点 是 root 是否可以获取 root 资源里面 index.php 这个后缀    }    // 直接解析URL地址    protected static function parseUrl($url, $domain)// 解析 你的 URL 跟 系统的解析 应该不一样吧    {        $request = Request::instance();// 获取 请求的 实例化        if (0 === strpos($url, '/')) {// 如果是 根目录形式的 直接解析            // 直接作为路由地址解析            $url = substr($url, 1);        } elseif (false !== strpos($url, '\\')) {// 这种写法 跟 类 命名空间相似            // 解析到类            $url = ltrim(str_replace('\\', '/', $url), '/');// 直接替换称为 url        } elseif (0 === strpos($url, '@')) {// 如果 有@ 符号,代表为控制器            // 解析到控制器            $url = substr($url, 1);        } else {            // 解析到 模块/控制器/操作            // 默认的大部分 应该就是这种情况            $module  = $request->module();// 获取模块 名字            $domains = Route::rules('domain');// 获取域名 规则            if (isset($domains[$domain]['[bind]'][0])) {// 如果关于这个规则 存在                $bindModule = $domains[$domain]['[bind]'][0];// 二级域名自动映射模块                if ($bindModule && !in_array($bindModule[0], ['\\', '@'])) {                    $module = '';// 模型为空                }            } else {                $module = $module ? $module . '/' : '';// 空模型            }            $controller = Loader::parseName($request->controller());// 控制器            if ('' == $url) {                // 空字符串输出当前的 模块/控制器/操作                $url = $module . $controller . '/' . $request->action();// 空的输出当前的            } else {// 重新 获取 这些                $path       = explode('/', $url);                $action     = Config::get('url_convert') ? strtolower(array_pop($path)) : array_pop($path);                $controller = empty($path) ? $controller : (Config::get('url_convert') ? Loader::parseName(array_pop($path)) : array_pop($path));                $module     = empty($path) ? $module : array_pop($path) . '/';                $url        = $module . $controller . '/' . $action;            }        }        return $url;// 返回 模型/控制器/方法    }