【发布时间】:2011-07-19 06:44:14
【问题描述】:
使用this question 的答案中的建议,以及提供几乎相同解决方案的this article,我一直在尝试设置 htaccess 来处理我的 js 和 css 文件的自动版本控制规则。
我想这样做的原因是我对它们进行了很多更改,但仍希望它们被浏览器缓存很长时间,而不必在每次更改时手动输入新的版本号。
使用的方法很简单:(1) 一个函数使用文件更改的日期将版本号附加到文件中,模式为[filename].[version_number].[suffix],因此style.css,例如,会变成@ 987654326@; (2) 使用 php,版本号包含在我网站页面的样式表声明中,如果版本化文件名与缓存的文件名不同,客户端浏览器将请求新副本; (3) .htaccess 中的 RewriteRule 使用 mod_rewrite 重写版本号,将其恢复为原始值并提供更新的文件。
下面列出了我在这三个阶段中的每一个阶段使用的代码。我正在http://edge.donaldjenkins.net/http://edge.donaldjenkins.net/博客的沙盒版本中的样式表上对此进行测试
1。在 WordPress 的 functions.php 文件中
// Allows autoversioning of css and js files
/**
* Given a file, i.e. /css/base.css, replaces it with a string containing the
* file's mtime, i.e. /css/base.1221534296.css.
*
* @param $file The file to be loaded. Must be an absolute path (i.e.
* starting with slash).
*/
function auto_version($file)
{
if(strpos($file, '/') !== 0 || !file_exists($_SERVER['DOCUMENT_ROOT'] . $file))
return $file;
$mtime = filemtime($_SERVER['DOCUMENT_ROOT'] . $file);
return preg_replace('{\\.([^./]+)$}', ".$mtime.\$1", $file);
}
2。在我的博客文件的<head> 部分中
<!-- Stylesheets -->
<link rel="stylesheet" href="<?=auto_version('/path/to/style.css')?>" />
3。在 .htaccess 文件中
# Allow versioning for js and css files
RewriteEngine On
RewriteRule ^(.*)\.[\d]+\.(css|js)$ $1.$2 [L] # Strip out the version number
# End Allow versioning for js and css files
通过上述设置,页面显示没有任何格式,并且生成页面的源代码显示样式表被引用为版本号——服务器上当然不存在。
这表明该函数正确地更改了样式表的名称,但由于某种原因,.htaccess 中 RewriteRule 中的正则表达式模式没有捕获文件名并重写它。事实上,如果我将其更改为^style.1300638388.css$ style.css [L],它甚至都不会捕获它。
我尝试了几种模式,但无济于事,但一定缺少一些非常基本的东西。
mod_rewrite 在服务器上,并且在其他几个 RewriteRule 实例上运行没有问题。
更新
.htaccessfile 中唯一的其他 RewriteRule 是标准的 WordPress 漂亮 url 重写规则。我怀疑它会干扰这个,但显然我无法在没有 WordPress 规则的情况下轻松测试,因为这会完全破坏页面生成:
# BEGIN WordPress
<IfModule mod_rewrite.c>
RewriteEngine On
RewriteBase /
RewriteRule ^index\.php$ - [L]
RewriteCond %{REQUEST_FILENAME} !-f
RewriteCond %{REQUEST_FILENAME} !-d
RewriteRule . /index.php [L]
</IfModule>
# END WordPress
更新 2:已解决
CharlesLeaf 在下面的评论中指出,由于 WordPress 规则在版本控制之前,因此不会执行后者。这实际上是原因,并允许我通过将两条规则分组来回答问题,如下所示:
<IfModule mod_rewrite.c>
# Allow versioning for js and css files
RewriteEngine On
RewriteRule ^(.*)\.[\d]+\.(css|js)$ $1.$2 [L] # Strip out the version number
# End Allow versioning for js and css files
# BEGIN WordPress
RewriteBase /
RewriteRule ^index\.php$ - [L]
RewriteCond %{REQUEST_FILENAME} !-f
RewriteCond %{REQUEST_FILENAME} !-d
RewriteRule . /index.php
# END WordPress
</IfModule>
【问题讨论】:
标签: php regex .htaccess mod-rewrite versioning