我正在建立一个相当大的网站,我的.htaccess开始感觉有点臃肿,有没有办法取代我现有的系统 - 每个可能传递的可能数量的一个规则,一个可以解释所有可以解释不同数量输入的表达式吗?
例如我目前有:
RewriteRule ^([a-z]+)/([^/]*)/([^/]*)/([^/]*)/([^/]*)/([^/]*)$ /index.php?mode=$1&id=$2&$3=$4&$5=$6
RewriteRule ^([a-z]+)/([^/]*)/([^/]*)/([^/]*)$ /index.php?mode=$1&id=$2&$3=$4
RewriteRule ^([a-z]+)/([^/]*)$ /index.php?mode=$1&id=$2
RewriteRule ^([a-z]+)$ /index.php?mode=$1
第一个反向引用始终是模式和(如果存在的话)第二个反向引用总是 id ,此后任何进一步的反向引用在输入名称和它之间交替值
http://www.example.com/search
http://www.example.com/search/3039/sort_by/name_asc/page/23
我希望能够有一个表达式来优雅地处理所有输入。
答案 0 :(得分:8)
像Drupal一样:
RewriteCond %{REQUEST_FILENAME} !-f
RewriteCond %{REQUEST_FILENAME} !-d
RewriteRule ^(.*)$ index.php?q=$1 [L,QSA]
然后使用像这样的PHP代码处理脚本中的所有内容
$pathmap = ();
if ($_GET["q"]){
$path = split("/", $_GET["q"]);
for ($i=0; $i+1<count($path); $i++){
$pathmap[$path[$i]] = $path[$i+1];
$i++;
}
}
答案 1 :(得分:8)
我不知道是否可以使用单个表达式完成,但无论查询字符串有多长,都可以使用固定数量的表达式完成。
您的mod_rewrite规则将被重复调用,为您提供有时称为mod_rewrite递归的内容。有避免它的技术,但我认为你想使用它。
设置一个规则,用name = value&amp;
替换最后一对继续将输入查询字符串添加到输出中。每次通过,您的查询字符串将变得更长,您的URL将变得更短。
最终,您只有一个与上一个规则匹配的值。
您必须使用
捕获查询字符串RewriteCond %{QUERY_STRING} ^(.*)$
然后使用%1
将其添加回输出你最终会有四行。
我知道您开始使用四行,但您可以根据需要匹配任意数量的参数,而无需添加第五行。
RewriteCond %{QUERY_STRING} ^(.*)$
RewriteRule ^(.*/)([^/]+)/([^/]+) $1?$2=$3&%1 [L]
RewriteCond %{QUERY_STRING} ^(.*)$
RewriteRule ^([^/]+)/ $1.php?%1 [L]
这将重写以下内容
/mypage/param1/val1/param2/val2/param3/val3/... --->
/mypage.php?param1=val1¶m2=val2¶m3=val3&...
只剩下一个参数时停止。它将采用第一个“参数”并使用该名称调用.php文件。 param / val对的数量没有限制。
答案 2 :(得分:2)
我不相信他们是一种方式 - 但我会说你最好的选择是让脚本“index.php”处理一条路径,而不是必须做很多反向引用。
例如,你的重写将是
RewriteBase /
RewriteCond %{REQUEST_FILENAME} !-f
RewriteCond %{REQUEST_FILENAME} !-d
RewriteRule . /index.php [L]
或者类似...这将使$ _SERVER ['REQUEST_URI']变量包含路径信息,您可以拆分并解析它。
$path = split('/', $_SERVER['REQUEST_URI']);
array_shift($path); // We always have a first null value
$mode = array_shift($path);
最终会出现包含模式的$ mode,以及包含路径其余部分的元素数组的$ path,所以
http://example.com/foo/bar/baz
让$ mode模式为'foo',$ path为包含'bar'和'baz'的数组