Redis 是一个高级的 key-value 存储系统,类似 memcached,所有内容都存在内存中,因此每秒钟可以超过 10 万次 GET 操作。
我下面提出的解决方案是在 Redis 中缓存所有输出的 HTML 内容而无需再让 WordPress 重复执行页面脚本。这里使用 Redis 代替 Varnish 设置更简单,速度更快。
如果你使用的是 Debian 或者衍生的操作系统可使用如下命令安装 Redis:
apt-get install redis-server
你需要一个客户端开发包以便 PHP 可以连接到 Redis 服务上。
这里我推荐 Predis。上传 predis.php 到 WordPress 的根目录。
index-with-redis.php
,内容如下:<?php
$cf = 1; // 如果使用 Cloudflare,设置为 1
$debug = 0; // 如果希望查看执行时间和缓存操作,设置为 1
$display_powered_by_redis = 1; // 如果希望显示 Redis 支持的消息和执行时间,设置为 1
$start = microtime(); // 开始计时页面执行时间
// 如果启用了 Cloudflare
if ($cf) {
if (isset($_SERVER['HTTP_CF_CONNECTING_IP'])) {
$_SERVER['REMOTE_ADDR'] = $_SERVER['HTTP_CF_CONNECTING_IP'];
}
}
// 来自 WordPress 的设置
define('WP_USE_THEMES', true);
// 初始化 Predis
include("predis.php");
$redis = new Predis\Client('');
// 初始化变量
$domain = $_SERVER['HTTP_HOST'];
$url = "http://" . $_SERVER['HTTP_HOST'] . $_SERVER['REQUEST_URI'];
$url = str_replace('?r=y', '', $url);
$url = str_replace('?c=y', '', $url);
$dkey = md5($domain);
$ukey = md5($url);
// 检查页面是否为评论提交
(isset($_SERVER['HTTP_CACHE_CONTROL']) && $_SERVER['HTTP_CACHE_CONTROL'] == 'max-age=0') ? $submit = 1 : $submit = 0;
// 检查是否已登录到 WordPress
$cookie = var_export($_COOKIE, true);
$loggedin = preg_match("/wordpress_logged_in/", $cookie);
// 检查页面缓存是否存在
if ($redis->hexists($dkey, $ukey) && !$loggedin && !$submit && !strpos($url, '/feed/')) {
echo $redis->hget($dkey, $ukey);
$cached = 1;
$msg = 'this is a cache';
// 如果提交了评论或发出清除页面缓存请求,删除页面缓存
} else if ($submit || substr($_SERVER['REQUEST_URI'], -4) == '?r=y') {
require('./wp-blog-header.php');
$redis->hdel($dkey, $ukey);
$msg = 'cache of page deleted';
// 如果登录用户发出清除整个缓存的请求,删除整个缓存
} else if ($loggedin && substr($_SERVER['REQUEST_URI'], -4) == '?c=y') {
require('./wp-blog-header.php');
if ($redis->exists($dkey)) {
$redis->del($dkey);
$msg = 'domain cache flushed';
} else {
$msg = 'no cache to flush';
}
// 如果已登录,不缓存任何内容
} else if ($loggedin) {
require('./wp-blog-header.php');
$msg = 'not cached';
// 缓存页面
} else {
// 开启输出缓冲
ob_start();
require('./wp-blog-header.php');
// 获取输出缓冲区的内容
$html = ob_get_contents();
// 清理输出缓冲区
ob_end_clean();
echo $html;
// 仅当页面存在且不是搜索结果时,才存储到缓存
if (!is_404() && !is_search()) {
// 将 HTML 内容存储到 Redis 缓存
$redis->hset($dkey, $ukey, $html);
$msg = 'cache is set';
}
}
$end = microtime(); // 获取执行时间的结束时间
// 如果启用了调试功能,显示消息
if ($debug) {
echo $msg . ': ';
echo t_exec($start, $end);
}
if ($cached && $display_powered_by_redis) {
// 您应该将此 CSS 移动到您的 CSS 文件并更改:float:right;margin:20px 0;
echo "<style>#redis_powered{float:right;margin:20px 0;background:url(http://images.staticjw.com/jim/3959/redis.png) 10px no-repeat #fff;border:1px solid #D7D8DF;padding:10px;width:190px;}
#redis_powered div{width:190px;text-align:right;font:10px/11px arial,sans-serif;color:#000;}</style>";
echo "<a href=\"http://www.aips.me/wordpress-with-redis-as-a-frontend-cache/\" style=\"text-decoration:none;\"><div id=\"redis_powered\"><div>Page generated in<br/> " . t_exec($start, $end) . " sec</div></div></a>";
}
// 计算时间差
function t_exec($start, $end) {
$t = (getmicrotime($end) - getmicrotime($start));
return round($t, 5);
}
// 获取时间
function getmicrotime($t) {
list($usec, $sec) = explode(" ", $t);
return ((float)$usec + (float)$sec);
}
?>
你也可以在 Github 上查看 index-with-redis.php
将上述代码中的 IP 地址和网站域名替换成你网站的 IP 地址和域名
在 .htaccess 中将所有出现 index.php
的地方改为 index-with-redis.php
,如果你使用的是 Nginx 则修改 nginx.conf 中的 index.php
为 index-with-redis.php
(并重载 Nginx:killall -s HUP nginx
)。
我已经在我的博客中使用了如上的方法进行加速很长时间了,一切运行良好。
本文作者的 WordPress 环境是 Nginx + PHP-FPM + APC + Cloudflare + Redis,安装在一个 VPS 中,无缓存插件。
请确认使用了 gzip 压缩,可加快访问速度。
要访问 wp-admin 必须使用 /wp-admin/index.php
代替原来的 /wp-admin/
本文其实在国内已经有很翻译过了,但我看到作者也一直在更新此文,反而国内译者都不怎么更新,我就自己去重新折腾了一遍。
英文原文:http://www.jimwestergren.com/wordpress-with-redis-as-a-frontend-cache/