说我的访客到达:
mydomain.com/nice-url/?p1=a\u0026amp;p2=b
我希望他们被重定向到:
mydomain.com/nice-url/?p1=a
(可能带头(“位置:)
所以我想剥离p2 = b字符串.
由于这只是一个例子,并且可能还有许多其他参数,我正在寻找最智能的解决方案,包括以下方法:
>剥去以p2开头的任何内容.
>剥去任何不以p1开头的东西
请记住,nice-url也可以是各种各样的东西,例如:
name-number-one,name-number-two等
另外,我不寻找.htaccess解决方案,因为我想在会话中存储剥离参数的值.
最佳答案 如建议你可以使用parse_url(
documentation).这是比hanshenrik发布的更通用的方式.它使用
documentation’s注释中的略微修改的unparse_url($query行被修剪为&)和一些正则表达式:
<?
function unparse_url($parsed_url) {
$scheme = isset($parsed_url['scheme']) ? $parsed_url['scheme'] . '://' : '';
$host = isset($parsed_url['host']) ? $parsed_url['host'] : '';
$port = isset($parsed_url['port']) ? ':' . $parsed_url['port'] : '';
$user = isset($parsed_url['user']) ? $parsed_url['user'] : '';
$pass = isset($parsed_url['pass']) ? ':' . $parsed_url['pass'] : '';
$pass = ($user || $pass) ? "$pass@" : '';
$path = isset($parsed_url['path']) ? $parsed_url['path'] : '';
$query = isset($parsed_url['query']) ? '?' . trim($parsed_url['query'], '&') : '';
$fragment = isset($parsed_url['fragment']) ? '#' . $parsed_url['fragment'] : '';
return "$scheme$user$pass$host$port$path$query$fragment";
}
function strip_query($url, $query_to_strip) {
$parsed = parse_url($url);
$parsed['query'] = preg_replace('/(^|&)'.$query_to_strip.'[^&]*/', '', $parsed['query']);
return unparse_url($parsed);
}
function keep_only_query($url, $query_to_keep) {
$parsed = parse_url($url);
$parsed['query'] = preg_replace('/(^|&)(?!'.$query_to_keep.')[^&]*/', '', $parsed['query']);
return unparse_url($parsed);
}
$url = 'mydomain.com/nice-url/?p1=a&p2=b';
# or for request url
# $url = "http://$_SERVER[HTTP_HOST]$_SERVER[REQUEST_URI]";
# "Strip anything that starts with p2"
var_dump(strip_query($url, 'p2'));
# --> string(27) "mydomain.com/nice-url/?p1=a"
# "Strip anything that doesn't start with p1"
var_dump(keep_only_query($url, 'p1'));
# --> string(27) "mydomain.com/nice-url/?p1=a"
# redirect
$new_url = strip_query($url, 'p2')); # or whatever query you want to strip/keep
header("Location: $new_url");
我希望keep_only_query正则表达式是完整的,并且适用于你遇到的所有网址,但这是我的第一个镜头.
也许正则表达式是矫枉过正(没有做任何基准测试),所以这里有两个函数的版本没有正则表达式,但有parse_str和http_build_query(由hanshendrik提出).
function strip_query($url, $query_to_strip) {
$parsed = parse_url($url);
parse_str($parsed['query'], $query);
unset($query[$query_to_strip]);
$parsed['query'] = http_build_query($query);
return unparse_url($parsed);
}
function keep_only_query($url, $query_to_keep) {
$parsed = parse_url($url);
parse_str($parsed['query'], $query);
foreach($query as $query_key => $query_val) {
if($query_key != $query_to_keep) unset($query[$query_key]);
}
$parsed['query'] = http_build_query($query);
return unparse_url($parsed);
}
注意:与正则表达式版本相反,这仅适用于具有完全名称的参数(例如p1,p2),而正则表达式版本适用于以名称开头的名称(例如p1 = …,p2 = …,但也p1something = …,p2_whatever = …).不知道问题的措辞是否意图,但这是我首先理解的方式:
- Strip anything that starts with …
当然,底部版本可以扩展为比较查询键的开头,也可以…与正则表达式;-)(或substr)
最后一个版本没有任何奇怪的正则表达式,但有substr,检查OP的意图的开头.
function strip_query($url, $query_to_strip) {
$len = strlen($query_to_strip);
$parsed = parse_url($url);
parse_str($parsed['query'], $query);
foreach($query as $query_key => $query_val) {
if(substr($query_key, 0, $len) == $query_to_strip) unset($query[$query_key]);
}
$parsed['query'] = http_build_query($query);
return unparse_url($parsed);
}
function keep_only_query($url, $query_to_keep) {
$len = strlen($query_to_keep);
$parsed = parse_url($url);
parse_str($parsed['query'], $query);
foreach($query as $query_key => $query_val) {
if(substr($query_key, 0, $len) != $query_to_keep) unset($query[$query_key]);
}
$parsed['query'] = http_build_query($query);
return unparse_url($parsed);
}