WordPress 使用 Redis 缓存来进行站点加速

Redis 是一个高级的 key-value 存储系统,类似 memcached,所有内容都存在内存中,因此每秒钟可以超过 10 万次 GET 操作。

我下面提出的解决方案是在 Redis 中缓存所有输出的 HTML 内容而无需再让 WordPress 重复执行页面脚本。这里使用 Redis 代替 Varnish 设置更简单,速度更快。

WordPress 使用 Redis 缓存来进行站点加速

安装 Redis

如果你使用的是 Debian 或者衍生的操作系统可使用如下命令安装 Redis:
apt-get install redis-server

使用 Predis 作为 Redis 的 PHP 客户端

你需要一个客户端开发包以便 PHP 可以连接到 Redis 服务上。

这里我推荐 Predis。上传 predis.php 到 WordPress 的根目录。

前端缓存的 PHP 脚本

  1. 在 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

  1. 将上述代码中的 IP 地址和网站域名替换成你网站的 IP 地址和域名

  2. 在 .htaccess 中将所有出现 index.php 的地方改为 index-with-redis.php,如果你使用的是 Nginx 则修改 nginx.conf 中的 index.phpindex-with-redis.php(并重载 Nginx:killall -s HUP nginx)。

性能测试

  • 没有 Redis 的情况下:平均首页执行 1.614 秒,文章页 0.174 秒(无任何缓存插件)
  • 使用 Redis 的情况下:平均页面执行时间 0.00256 秒

我已经在我的博客中使用了如上的方法进行加速很长时间了,一切运行良好。

其他建议

本文作者的 WordPress 环境是 Nginx + PHP-FPM + APC + Cloudflare + Redis,安装在一个 VPS 中,无缓存插件。

请确认使用了 gzip 压缩,可加快访问速度。

访问 wp-admin

要访问 wp-admin 必须使用 /wp-admin/index.php 代替原来的 /wp-admin/

本文其实在国内已经有很翻译过了,但我看到作者也一直在更新此文,反而国内译者都不怎么更新,我就自己去重新折腾了一遍。
英文原文:http://www.jimwestergren.com/wordpress-with-redis-as-a-frontend-cache/