php parse_url reverse - 解析url

时间:2010-12-04 17:49:15

标签: php parse-url

有没有办法从解析后的网址中撤消网址?

$url = 'http://www.domain.com/dir/index.php?query=blabla#more_bla';
$parse = parse_url($url);
print_r($parse);
/*
array(
 'scheme'=>'http://',
 etc....
)
*/
$revere = reverse_url($parse); // probably does not exist but u get the point

echo $reverse;
//outputs:// "http://www.domain.com/dir/index.php?query=blabla#more_bla"

或者,如果有办法验证丢失了部分推荐网址的网址,例如

www.mydomain.com

mydomain.com

应该全部归来 http://www.mydomain.com 或者使用正确的子域

7 个答案:

答案 0 :(得分:20)

这些是我用于分解和重建URL的两个函数:

function http_parse_query($query) {
    $parameters = array();
    $queryParts = explode('&', $query);
    foreach ($queryParts as $queryPart) {
        $keyValue = explode('=', $queryPart, 2);
        $parameters[$keyValue[0]] = $keyValue[1];
    }
    return $parameters;
}

function build_url(array $parts) {
    return (isset($parts['scheme']) ? "{$parts['scheme']}:" : '') . 
        ((isset($parts['user']) || isset($parts['host'])) ? '//' : '') . 
        (isset($parts['user']) ? "{$parts['user']}" : '') . 
        (isset($parts['pass']) ? ":{$parts['pass']}" : '') . 
        (isset($parts['user']) ? '@' : '') . 
        (isset($parts['host']) ? "{$parts['host']}" : '') . 
        (isset($parts['port']) ? ":{$parts['port']}" : '') . 
        (isset($parts['path']) ? "{$parts['path']}" : '') . 
        (isset($parts['query']) ? "?{$parts['query']}" : '') . 
        (isset($parts['fragment']) ? "#{$parts['fragment']}" : '');
}

// Example
$parts = parse_url($url);

if (isset($parts['query'])) {
    $parameters = http_parse_query($parts['query']);
    foreach ($parameters as $key => $value) {
        $parameters[$key] = $value; // do stuff with $value
    }
    $parts['query'] = http_build_query($parameters);
}

$url = build_url($parts);

答案 1 :(得分:18)

你应该能够做到

http_build_url($parse)

注意:http_build_url仅在安装pecl_http时可用。

根据文档,它专门用于处理parse_url的输出。这两个函数都处理锚点,查询参数等,所以没有“$ url上没有提到的其他属性”。

要删除http://,请在解析之前使用基本检查:

if (strpos($url, "http://") != 0)
    $url = "http://$url";

答案 2 :(得分:7)

这个功能可以解决问题:

/**
 * @param array $parsed
 * @return string
 */
function unparse_url(array $parsed) {
    $get = function ($key) use ($parsed) {
        return isset($parsed[$key]) ? $parsed[$key] : null;
    };

    $pass      = $get('pass');
    $user      = $get('user');
    $userinfo  = $pass !== null ? "$user:$pass" : $user;
    $port      = $get('port');
    $scheme    = $get('scheme');
    $query     = $get('query');
    $fragment  = $get('fragment');
    $authority =
        ($userinfo !== null ? "$userinfo@" : '') .
        $get('host') .
        ($port ? ":$port" : '');

    return
        (strlen($scheme) ? "$scheme:" : '') .
        (strlen($authority) ? "//$authority" : '') .
        $get('path') .
        (strlen($query) ? "?$query" : '') .
        (strlen($fragment) ? "#$fragment" : '');
}

这是一个简短的测试:

function unparse_url_test() {
    foreach ([
        '',
        'foo',
        'http://www.google.com/',
        'http://u:p@foo:1/path/path?q#frag',
        'http://u:p@foo:1/path/path?#',
        'ssh://root@host',
        '://:@:1/?#',
        'http://:@foo:1/path/path?#',
        'http://@foo:1/path/path?#',
    ] as $url) {
        $parsed1 = parse_url($url);
        $parsed2 = parse_url(unparse_url($parsed1));

        if ($parsed1 !== $parsed2) {
            print var_export($parsed1, true) . "\n!==\n" . var_export($parsed2, true) . "\n\n";
        }
    }
}

unparse_url_test();

答案 3 :(得分:1)

另一个实施:

function build_url(array $elements) {
    $e = $elements;
    return
        (isset($e['host']) ? (
            (isset($e['scheme']) ? "$e[scheme]://" : '//') .
            (isset($e['user']) ? $e['user'] . (isset($e['pass']) ? ":$e[pass]" : '') . '@' : '') .
            $e['host'] .
            (isset($e['port']) ? ":$e[port]" : '')
        ) : '') .
        (isset($e['path']) ? $e['path'] : '/') .
        (isset($e['query']) ? '?' . (is_array($e['query']) ? http_build_query($e['query'], '', '&') : $e['query']) : '') .
        (isset($e['fragment']) ? "#$e[fragment]" : '')
    ;
}

结果应该是:

{
    "host": "example.com"
}
/* //example.com/ */

{
    "scheme": "https",
    "host": "example.com"
}
/* https://example.com/ */

{
    "scheme": "http",
    "host": "example.com",
    "port": 8080,
    "path": "/x/y/z"
}
/* http://example.com:8080/x/y/z */

{
    "scheme": "http",
    "host": "example.com",
    "port": 8080,
    "user": "anonymous",
    "query": "a=b&c=d",
    "fragment": "xyz"
}
/* http://anonymous@example.com:8080/?a=b&c=d#xyz */

{
    "scheme": "http",
    "host": "example.com",
    "user": "root",
    "pass": "stupid",
    "path": "/x/y/z",
    "query": {
        "a": "b",
        "c": "d"
    }
}
/* http://root:stupid@example.com/x/y/z?a=b&c=d */

{
    "path": "/x/y/z",
    "query": "a=b&c=d"
}
/* /x/y/z?a=b&c=d */

答案 4 :(得分:1)

此答案是@BradMace接受的答案的附录。我最初将其添加为评论,但他建议将其添加为单独的答案,所以就在这里。

http_build_url($parse)提供的使用pecl_http的原始答案适用于扩展版本1.x-版本2.x及更高版本是面向对象的,并且语法已更改。

在较新版本中(在pecl_http v.3.2.3上测试),实现应为:

$httpUrl = new \http\Url($parsed);
$url = $httpUrl->toString();

答案 5 :(得分:0)

十年后,使用数十年的方法:)

考虑到您将永远有计划和房东。 (可选)路径,查询和片段偏移:

$a = parse_url('https://example.com/whatever-season/?drink=water&sleep=better');

使用经典的sprintf重新构建:

function buildUrl(array $a){
        return sprintf('%s://%s%s%s%s',
                            $a['scheme'], $a['host'], $a['path'] ?? '',
                            $a['query'] ? '?' . $a['query'] : '',
                            $a['fragment'] ? '#' . $a['fragment'] : '');
}

其中{strong> path 偏移量由the null coalescing operator(PHP 7+)操作。

答案 6 :(得分:-1)

从网址获取最后一个字符串 例如:http://example.com/controllername/functionname 并需要获取functionname

$ referer = explode('/',strrev($ _ SERVER ['HTTP_REFERER']));

$ lastString = strrev($ referer [0]);