请选择 进入手机版 | 继续访问电脑版
查看: 69|回复: 0

Discuz缓存机制详解

[复制链接]

494

主题

496

帖子

2576

积分

管理员

Rank: 9Rank: 9Rank: 9

积分
2576
发表于 2021-6-4 12:59:41 | 显示全部楼层 |阅读模式
一、DICUZ缓存机制分析:

使用缓存机制的目的就是,降低服务器性能的消耗,对于常用且变动比较小的数据,可以尽可能的使用缓存来解决,代替最原生的不断的进行数据库查询匹配的过程。而Discuz中提供的数据缓存方式包括内存、数据库、文件三种方式,具体详解如下。

(1)、项目根目录/config/config_global.php配置数据库/文件缓存方式,当然还有第三种,就是内存缓存。
  1. $_config['cache']['type'] = 'sql';//file
复制代码
(2)、缓存数据获取:
根目录/source/function/function_core.php:

  1. function loadcache($cachenames, $force = false) {
  2.     global $_G;
  3.     static $loadedcache = array();
  4.     $cachenames = is_array($cachenames) ? $cachenames : array($cachenames);
  5.     $caches = array();
  6.     foreach ($cachenames as $k) {
  7.         if(!isset($loadedcache[$k]) || $force) {
  8.             $caches[] = $k;
  9.             $loadedcache[$k] = true;
  10.         }
  11.     }

  12.     if(!empty($caches)) {
  13.         $cachedata = C::t('common_syscache')->fetch_all($caches);
  14.         foreach($cachedata as $cname => $data) {
  15.             if($cname == 'setting') {
  16.                 $_G['setting'] = $data;
  17.             } elseif($cname == 'usergroup_'.$_G['groupid']) {
  18.                 $_G['cache'][$cname] = $_G['group'] = $data;
  19.             } elseif($cname == 'style_default') {
  20.                 $_G['cache'][$cname] = $_G['style'] = $data;
  21.             } elseif($cname == 'grouplevels') {
  22.                 $_G['grouplevels'] = $data;
  23.             } else {
  24.                 $_G['cache'][$cname] = $data;
  25.             }
  26.         }
  27.     }
  28.     return true;
  29. }
复制代码
从函数中可以看出,数据缓存其实相当于把一些常用的数据,通过特定的需求,根据缓存的方式存储于文件 或者 表(前缀)_common_syscache 或者 内存中,当需要使用到某个类型的数据的时候,只需要在代码中加入类似loadcache('setting');,就可以获取到缓存数据并赋值于自定义全局变量$_G中,即$_G['setting']; ,其中fetch_all中会判断当前使用的是哪一种缓存方式,如下该函数:

脚本:table_common_syscache.php

  1. public function fetch_all($cachenames) {
  2.         $data = array();
  3.         $cachenames = is_array($cachenames) ? $cachenames : array($cachenames);
  4.         if($this->_allowmem) {
  5.             $data = memory('get', $cachenames);
  6.             $newarray = $data !== false ? array_diff($cachenames, array_keys($data)) : $cachenames;
  7.             if(empty($newarray)) {
  8.                 return $data;
  9.             } else {
  10.                 $cachenames = $newarray;
  11.             }
  12.         }

  13.         if($this->_isfilecache) {
  14.             $lostcaches = array();
  15.             foreach($cachenames as $cachename) {
  16.                 if(!@include_once(DISCUZ_ROOT.'./data/cache/cache_'.$cachename.'.php')) {
  17.                     $lostcaches[] = $cachename;
  18.                 } elseif($this->_allowmem) {
  19.                     memory('set', $cachename, $data[$cachename]);
  20.                 }
  21.             }
  22.             if(!$lostcaches) {
  23.                 return $data;
  24.             }
  25.             $cachenames = $lostcaches;
  26.             unset($lostcaches);
  27.         }

  28.         $query = DB::query('SELECT * FROM '.DB::table($this->_table).' WHERE '.DB::field('cname', $cachenames));
  29.         while($syscache = DB::fetch($query)) {
  30.             $data[$syscache['cname']] = $syscache['ctype'] ? unserialize($syscache['data']) : $syscache['data'];
  31.             $this->_allowmem && (memory('set', $syscache['cname'], $data[$syscache['cname']]));
  32.             if($this->_isfilecache) {
  33.                 $cachedata = '$data[\''.$syscache['cname'].'\'] = '.var_export($data[$syscache['cname']], true).";\n\n";
  34.                 if(($fp = @fopen(DISCUZ_ROOT.'./data/cache/cache_'.$syscache['cname'].'.php', 'wb'))) {
  35.                     fwrite($fp, "<?php\n//Discuz! cache file, DO NOT modify me!\n//Identify: ".md5($syscache['cname'].$cachedata.getglobal('config/security/authkey'))."\n\n$cachedata?>");
  36.                     fclose($fp);
  37.                 }
  38.             }
  39.         }

  40.         foreach($cachenames as $name) {
  41.             if($data[$name] === null) {
  42.                 $data[$name] = null;
  43.                 $this->_allowmem && (memory('set', $name, array()));
  44.             }
  45.         }

  46.         return $data;
  47. }
复制代码
当服务器支持使用内存缓存时,使用该方式,DISCUZ代码中默认支持的方式包括redis、memcache、apc、xcache、eaccelerator、wincache这几种,至于使用哪一种,还需要取决于web 服务器支持方式,获取到数据且不空则返回其结果;

当服务器不支持内存缓存时,使用第二种文件缓存,文件缓存会将数据存放于目录./data/cache/ 下,并取名为cache_'.$cachename.'.php  ,如果获取到数据且不空,则返回其结果;

而数据库缓存查找操作是在前面两种不支持,或者执行完且数据为空的情况下,根据库缓存从数据库表(前缀)_common_syscache中获取存储数据,其实这里就不能描述为缓存,只是把数据按照一定的格式存放在数据库中,由于其使用其数据的客户端不需要经常实时变动,也就是说可以存在一定的延迟,才使用这样的方式,简化数据库操作,存储方式包括字段缓存变量名(如setting)、缓存类型(0,1是否为序列化数据)、缓存时间、缓存数据。

当执行了数据库查询操作之后,如果检测到存在内存缓存or文件缓存,由于从前面逻辑可以判断,如果支持该两种缓存方式,能够执行数据库查询操作是因为该两种缓存方式缓存的数据为空,这个时候就需要将数据库查询到的数据缓存到内存缓存或者文件缓存中。(这里就是文件缓存数据生成的位置,而数据缓存生成的数据则存放与./data/cache/目录下)


(3)、缓存数据生成:(文件缓存数据根据数据是否需要缓存,且支持文件缓存,会在数据获取的过程中存储于缓存文件中,缓存文件中如果存在数据的话,则不会从数据库中获取数据,具体详情如“缓存数据获取”中有所描述)

数据库缓存生成:

方式一:
根目录/source/function/function_core.php,直接数据库缓存数据的存储:
  1. function savecache($cachename, $data) {
  2.     C::t('common_syscache')->insert($cachename, $data);
  3. }

  4. function save_syscache($cachename, $data) {
  5.     savecache($cachename, $data);
  6. }
复制代码
很简单,把需要缓存的数据直接存储于需要的位置中(内存、文件、数据库),如下insert操作:
  1. public function insert($cachename, $data) {

  2.         parent::insert(array(
  3.             'cname' => $cachename,
  4.             'ctype' => is_array($data) ? 1 : 0,
  5.             'dateline' => TIMESTAMP,
  6.             'data' => is_array($data) ? serialize($data) : $data,
  7.         ), false, true);

  8.         if($this->_allowmem && memory('get', $cachename) !== false) {
  9.             memory('set', $cachename, $data);
  10.         }
  11.         $this->_isfilecache && @unlink(DISCUZ_ROOT.'./data/cache/cache_'.$cachename.'.php');
  12. }
复制代码
存储数据将存放于缓存数据库中,内存中,以及删除原始的缓存文件(目的请参照缓存数据获取时文件缓存的生成过程,当文件缓存数据为空时,会重新生成文件缓存数据)

方式二:

即使用function_cache.php脚本中的updatecache($cachename)函数,该函数的实现机制协同项目目录./source/function/cache/cache_*.php脚本。

而./source/function/cache/cache_*.php脚本最终的依然是调用save_syscache函数,所以,为了方便数据格式化,在需要存储特定数据的时候,可以在./source/function/cache/目录下创建相应的数据生产脚本文件,该脚本会通过原有的逻辑,通过调用updatecache('特定标示符') 函数,然后调用cache_特定标示符.php脚本生成缓存数据,缓存数据的存放根据系统设定好的缓存方式进行存储。

二、Discuz官方提供“Discuz缓存机制”——原文:http://bbs.zb7.com/discuz/dx25/plug/construct/construct_cache.htm

Discuz! X2.5 的 config_global.php 中有这样一行代码

$_config['cache']['type'] = 'sql';

这就是 Discuz! 内置的缓存方式,如果填写 'sql' 则为使用数据库缓存,填写 'file' 则为使用文件缓存。

内存级缓存

缓存层的引入是为了解决MYSQL自身对高并发处理的性能瓶颈,目前产品缓存层采用主流的Key-Value对形式,内存级的缓存产品很多,支持的内存优化接口有 Memcache、eAccelerator、Alternative PHP Cache(APC)、Xcache、Redis 五种,优化系统将会依据当前服务器环境依次选用接口,单服务器环境中推荐使用APC,多服务器环境中推荐使用Redis或Memcache。

数据层是以表为单位的类文件,所有表类都继承discuz_table基类,基类实现缓存操作的相关函数;理论上所有的数据表均可以缓存,目前产品在六个压力大的数据表内置开启了缓存机制:用户相关表、回帖、主题、主题和专辑关系、淘贴专辑、用户关注关系。

用户相关表
缓存表:'common_member', 'count', 'status','profile', 'field_home', 'field_forum'。

UID为缓存KEY。表数据更新时缓存数据会同步更新。

回帖
以TID为单位,缓存第一页的post数据。表数据更新时缓存数据会同步更新。

主题
TID为缓存KEY。表数据更新时缓存数据会同步更新。

版块列表默认参数第一页时以 forumdisplay_FID 为缓存KEY,缓存时间内数据不更新。

主题和专辑关系
以TID为单位,此TID的专辑ID集合,表数据更新时缓存数据会同步更新。

淘贴专辑
以TID为单位,此TID的专辑集合,缓存时间内数据不更新。

用户关注关系
以UID为单位,此UID关注用户的关系数据,缓存时间内数据不更新。

内存级缓存层实现细节

discuz_table基类中缓存机制的实现
  1. /**
  2. * @var string 缓存主键名前缀,为空时表示此表不支持缓存
  3. */
  4. protected $_pre_cache_key;

  5. /**
  6. * @var string 缓存时间,以秒为单位,0表示永久或相关配置文件中的默认值
  7. */
  8. protected $_cache_ttl;
复制代码
discuz_table基类中缓存机制的方法
  1. //缓存一个变量到缓存中,如果 KEY已经在则会被覆盖为新值
  2. store_cache($id, $data, $cache_ttl = null, $pre_cache_key = null)
  3. //获取指定KEY的缓存数据
  4. fetch_cache($ids, $pre_cache_key = null)
  5. //清除指定KEY的缓存
  6. clear_cache($ids, $pre_cache_key = null)
  7. //更新一个已经存在的KEY,只更新修改的字段
  8. update_cache($id, $data, $cache_ttl = null, $pre_cache_key = null)
  9. //批量更新缓存,只更新已经存在KEY的指定修改的字段
  10. update_batch_cache($ids, $data, $cache_ttl = null, $pre_cache_key = null)
  11. //重置已经存在的KEY的值
  12. reset_cache($ids, $pre_cache_key = null)
  13. //累加缓存数据中某字段的值
  14. increase_cache($ids, $data, $cache_ttl = null, $pre_cache_key = null)
复制代码


回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

QQ|Archiver|手机版|小黑屋|蓄财网 ( 豫ICP备2021002293号 )

GMT+8, 2021-6-17 02:00 , Processed in 0.087319 second(s), 33 queries .

蓄财网 ©版权所有

© 2021 蓄财网(www.aixc.cn)

快速回复 返回顶部 返回列表