Discuz!下Memcache缓存实现方法 |
本文标签:Discuz,Memcache 前言: 系统分析: 就我手头的一些数据显示,目前的Discuz!论坛的基本服务器架构是前面Squid顶着,后面才是一台DB在撑着 。这种架构中,web服务器压力增大可以通过并行增加服务器解决,而MySQL压力却无处释放,在不考虑MySQL官方服务的情况下,我们通过合理的利用Memcache是可以达到减轻MySQL服务器负载的 。 可能会有朋友说我们可以对数据表进行分表(注:此处分表是指通过PHP程序去分表,比如pw,dv的分表)处理,但是当前的情况是一台DB服务器已经不能支撑当前的数据处理了,通过PHP对MySQL进行的分表依然不能减轻MySQL的负载 。(注:本段文字针对已经成型的系统,如果是独立开发的系统在架构前期就进行数据的同步分区还是不错的 。) 还可能有朋友会说利用MySQL的主从构架,如果你提出这个问题,我就很明确的告诉你,回去看看手册吧 。在Mysql Master/Slave 模式中,Slave主要是来备份数据的,只有当Master出现故障时,Slave才会接过Master的服务,对外部请求进行处理,直到Master恢复正常 。就是说:在Master/Slave中,要么是Master在服务,要么是Slave在服务,不会Master/Slave同时提供服务 。使用MySQL主从依然不能有效的降低MySQL的负载 。 或许你又会问我为什么不使用MySQL集群(MySQL Cluster),那可是白花花的银子啊,同等金钱的付出下,获得最大的收益才是王道 。PS:说句题外话,MySQL手册中将MySQL集群解释为MySQL簇,不习惯 。 其实在MySQL5.1中的MySQL分区(MySQL Partition)是个很好的东西,它允许根据可以设置为任意大小的规则,跨文件系统分配单个表的多个部分 。实际上,表的不同部分在不同的位置被存储为单独的表 。我认为这个才是当前情况下,最积极有效的降低MySQL负载的解决方法之一 。但是遗憾的是,这种MySQL分区的方式我个人没有使用过的经历,也不见有相当充分的案例表明它是稳定的或者不稳定的 。所以我还在徘徊中 。如果你知道,请麻烦告之!有朋友说腾讯是在用MySQL分区,但是遗憾的是我没有得到确切的数据 。 好了分析总结了这么多种降低MySQL负载的方式之后,在用户环境需求等特定条件下,我得出结论在当前情况下,缓解Discuz!论坛的MySQL负载比较有效的方法就是使用Memcache!
使用Memcache的理由: Discuz!使用Memcache $memcachehost = 127.0.0.1; $memcacheport = 11211; $memcachelife = 60; 2.在include/common.inc.php中 $mem = new Memcache; $mem->connect($memcachehost, $memcacheport); 3.修改include/db_mysql.class.php中的fetch_array、query这两个方法,并添加query_mysql方法,代码如下: function fetch_array($query, $result_type = MYSQL_ASSOC) { return is_resource($query) ? mysql_fetch_array($query, $result_type) : $query[0]; } function query_memcache($sql, $type = ) { global $mem,$memcachelife; $key = md5($sql); if(!($query = $mem->get($key))) { $query = $this->query($sql, $type); while($item = $this->fetch_array($query)) { $res[] = $item; } $query = $res; $mem->set($key, $query , 0, $memcachelife); } return $query; } function query($sql, $type = ) { global $debug, $discuz_starttime, $sqldebug, $sqlspenttimes; $func = $type == UNBUFFERED && @function_exists(mysql_unbuffered_query) ? mysql_unbuffered_query : mysql_query; if(!($query = $func($sql, $this->link)) && $type != SILENT) { $this->halt(MySQL Query Error, $sql); } if(substr($sql, 0, 6) == SELECT) { echo <font color="red">Cache SQL</font>:<font color="green">.$sql.</font><br /><br />; } else { echo <font color="red">Flash SQL</font>:<font color="green">.$sql.</font><br /><br />; } $this->querynum++; return $query; } 4.将需要使用Memcache缓存的SQL查询的代码由 $db->query( 修改为 $db->query_memcache( 注意并将 while($post = $db->fetch_array($query)) { 修改为 foreach($query as $post) { 没有while的$db->fetch_array可以不用修改 。 下面代码有用得着的就拿去: preg_replace("/while\([$](\w+)\s*\=\s*[$]db->fetch_array\([$]query\)\)/is", "foreach(\$query as \$\\1)", $file); 回头放出个小工具批量替换下就可以了 。 |