设为首页 收藏本站
查看: 1091|回复: 0

[经验分享] Discuz!NT中的Redis架构设计

[复制链接]

尚未签到

发表于 2015-7-19 12:14:32 | 显示全部楼层 |阅读模式
在之前的Discuz!NT缓存的架构方案中,曾说过Discuz!NT采用了两级缓存方式,即本地缓存+memcached方式。在近半年多的实际运行环境下,该方案经受住了检验。现在为了提供多样式的解决方案,我在企业版里引入了Redis这个目前炙手可热的缓存架构产品,即将memcached与Redis作为可选插件方式来提供了最终用户,尽管目前测试的结果两者的差异不是很大(毫秒级),但我想多一种选择对用户来说也是好的。
     闲话不多说了,开始今天的正文吧。
   
     熟悉我们产品的开发者都知道,我们的产品在缓存设计上使用了策略模式(Stratety Pattern),之前在系统中就已实现了DefaultCacheStrategy和MemCachedStrategy,前者用于本地缓存方式,后者顾名思义,就是memcached缓存方式。所以只要我们实现一个RedisStrategy策略,并适当修改加载缓存策略的代码,就可以将memcached缓存方式替换成Redis,如下图:
       DSC0000.png

     下面我先将RedisStrategy的部分代码放上来,大家一看便知:


DSC0001.gif DSC0002.gif

///
/// 企业级Redis缓存策略类
///
public class RedisStrategy : DefaultCacheStrategy
{
    ///
    /// 添加指定ID的对象
    ///
    ///
    ///
    public override void AddObject(string objId, object o)
    {  
        if (!objId.StartsWith("/Forum/ShowTopic/"))
            base.AddObject(objId, o, LocalCacheTime);
        using (IRedisClient Redis = RedisManager.GetClient())
        {
            Redis.Set(objId, new ObjectSerializer().Serialize(o));
        }
    }
    ///
    /// 加入当前对象到缓存中
    ///
    /// 对象的键值
    /// 缓存的对象
    /// 到期时间,单位:秒
    public override void AddObject(string objId, object o, int expire)
    {
        //凡是以"/Forum/ShowTopic/"为前缀不添加到本地缓存中,现类似键值包括: "/Forum/ShowTopic/Tag/{topicid}/" , "/Forum/ShowTopic/TopList/{fid}"
        if (!objId.StartsWith("/Forum/ShowTopic/"))
            base.AddObject(objId, o, expire);
        using (IRedisClient Redis = RedisManager.GetClient())
        {
            //永不过期
            if (expire == 0)
                Redis.Set(objId, new ObjectSerializer().Serialize(o));
            else
                Redis.Set(objId, new ObjectSerializer().Serialize(o), DateTime.Now.AddSeconds(expire));
        }         
   }

    ///
    /// 移除指定ID的对象
    ///
    ///
    public override void RemoveObject(string objId)
    {
        //先移除本地cached,然后再移除memcached中的相应数据
        base.RemoveObject(objId);
        using (IRedisClient Redis = RedisManager.GetClient())
        {
            Redis.Remove(objId);
        }
        Discuz.EntLib.SyncCache.SyncRemoteCache(objId);
    }      
    public override object RetrieveObject(string objId)
    {
        object obj = base.RetrieveObject(objId);
        if (obj == null)
        {
            using (IRedisClient Redis = RedisManager.GetClient())
            {
                obj = new ObjectSerializer().Deserialize(Redis.Get(objId));
                if (obj != null && !objId.StartsWith("/Forum/ShowTopic/"))//对ShowTopic页面缓存数据不放到本地缓存
                {
                    if (objId.StartsWith("/Forum/ShowTopicGuestCachePage/"))//对游客缓存页面ShowTopic数据缓存设置有效时间
                        base.TimeOut = GeneralConfigs.GetConfig().Guestcachepagetimeout * 60;
                    if (objId.StartsWith("/Forum/ShowForumGuestCachePage/"))//对游客缓存页面ShowTopic数据缓存设置有效时间
                        base.TimeOut = RedisConfigs.GetConfig().CacheShowForumCacheTime * 60;
                    else
                        base.TimeOut = LocalCacheTime;
                    base.AddObject(objId, obj, TimeOut);
                }               
            }
        }
        return obj;
    }
    ///
    /// 到期时间,单位:秒
    ///
    public override int TimeOut
    {
        get
        {
            return 3600;
        }
    }
    ///
    /// 本地缓存到期时间,单位:秒
    ///
    public int LocalCacheTime
    {
        get
        {
            return RedisConfigs.GetConfig().LocalCacheTime;
        }
    }
    ///
    /// 清空的有缓存数据
    ///
    public override void FlushAll()
    {
        base.FlushAll();
        using (IRedisClient Redis = RedisManager.GetClient())
        {
            Redis.FlushAll();
        }
    }
}
     可以看出RedisStrategy类继承自DefaultCacheStrategy,这一点与MemCachedStrategy实现如出一辙,唯一不同是其缓存数据加载和设置的方式有所不同,而具体的用法可以参见我之前写的这篇文章中的“object序列化方式存储”  。
   
     当然上面代码中有两个类未说明,一个是RedisConfigs,一个是RedisManager,前者是配置文件信息类,我们产品因为使用了统一的序列化接口实现方式(参见该文),所以其实现方式比较清晰,其序列化类的结构如下:



///
/// Redis配置信息类文件
///
public class RedisConfigInfo : IConfigInfo
{
    private bool _applyRedis;
    ///
    /// 是否应用Redis
    ///
    public bool ApplyRedis
    {
        get
        {
            return _applyRedis;
        }
        set
        {
            _applyRedis = value;
        }
    }
    private string _writeServerList;
    ///
    /// 可写的Redis链接地址
    ///
    public string WriteServerList
    {
        get
        {
            return _writeServerList;
        }
        set
        {
            _writeServerList = value;
        }
    }
    private string _readServerList;
    ///
    /// 可读的Redis链接地址
    ///
    public string ReadServerList
    {
        get
        {
            return _readServerList;
        }
        set
        {
            _readServerList = value;
        }
    }
    private int _maxWritePoolSize;
    ///
    /// 最大写链接数
    ///
    public int MaxWritePoolSize
    {
        get
        {
            return _maxWritePoolSize > 0 ? _maxWritePoolSize : 5;
        }
        set
        {
            _maxWritePoolSize = value;
        }
    }
    private int _maxReadPoolSize;
    ///
    /// 最大读链接数
    ///
    public int MaxReadPoolSize
    {
        get
        {
            return _maxReadPoolSize > 0 ? _maxReadPoolSize : 5;
        }
        set
        {
            _maxReadPoolSize = value;
        }
    }
    private bool _autoStart;
    ///
    /// 自动重启
    ///
    public bool AutoStart
    {
        get
        {
            return _autoStart;
        }
        set
        {
            _autoStart = value;
        }
    }
   
    private int _localCacheTime = 30000;
    ///
    /// 本地缓存到期时间,该设置会与memcached搭配使用,单位:秒
    ///
    public int LocalCacheTime
    {
        get
        {
            return _localCacheTime;
        }
        set
        {
            _localCacheTime = value;
        }
    }
    private bool _recordeLog = false;
    ///
    /// 是否记录日志,该设置仅用于排查redis运行时出现的问题,如redis工作正常,请关闭该项
    ///
    public bool RecordeLog
    {
        get
        {
            return _recordeLog;
        }
        set
        {
            _recordeLog = value;
        }
    }
    private int _cacheShowTopicPageNumber = 5;
    ///
    /// 缓存帖子列表分页数(showtopic页数使用缓存前N页的帖子列表信息)
    ///
    public int CacheShowTopicPageNumber
    {
        get
        {
            return _cacheShowTopicPageNumber;
        }
        set
        {
            _cacheShowTopicPageNumber = value;
        }
    }
    ///
    /// 缓存showforum页面分页数
    ///
    public int CacheShowForumPageNumber{set;get;}
    ///
    /// 缓存showforum页面时间(单位:分钟)
    ///
    public int CacheShowForumCacheTime{set;get;}
}   
     其序列化出来的xml文件格式形如:
   





  true
  10.0.4.210:6379
  10.0.4.210:6379
  60
  60
  true
  180
  
  false
  
  2
  
  2
  
  10

     之前所说的RedisManager类则是一个RedisClient客户端实现的简单封装,主要为了简化基于链接池的Redis链接方式的使用。其结构如下:



using System.Collections;
using Discuz.Config;
using Discuz.Common;
using ServiceStack.Redis;
using ServiceStack.Redis.Generic;
using ServiceStack.Redis.Support;
namespace Discuz.EntLib
{
    ///
    /// MemCache管理操作类
    ///
    public sealed class RedisManager
    {
        ///
        /// redis配置文件信息
        ///
        private static RedisConfigInfo redisConfigInfo = RedisConfigs.GetConfig();
        private static PooledRedisClientManager prcm;
        ///
        /// 静态构造方法,初始化链接池管理对象
        ///
        static RedisManager()
        {
            CreateManager();
        }

        ///
        /// 创建链接池管理对象
        ///
        private static void CreateManager()
        {
            string[] writeServerList = Utils.SplitString(redisConfigInfo.WriteServerList, ",");
            string[] readServerList = Utils.SplitString(redisConfigInfo.ReadServerList, ",");
            prcm = new PooledRedisClientManager(readServerList, writeServerList,
                             new RedisClientManagerConfig
                             {
                                 MaxWritePoolSize = redisConfigInfo.MaxWritePoolSize,
                                 MaxReadPoolSize = redisConfigInfo.MaxReadPoolSize,
                                 AutoStart = redisConfigInfo.AutoStart,
                             });           
        }
        ///
        /// 客户端缓存操作对象
        ///
        public static IRedisClient GetClient()
        {
            if (prcm == null)
                CreateManager();
            return prcm.GetClient();
        }
    }
}
     上面的代码主要将redis.config的配置文件文件信息加载到程序里并实始化PooledRedisClientManager对象,该对象用于池化redis的客户端链接,具体方式参见这篇文章
        
      好了,到这里主要的内容就介绍完了。
     
      注:本文的部分代码位于企业版产品中,目前暂未开源所以大家可能没有拿到,我们计划今年开源企业版1.0的代码,包括本文中代码部分,以便从社区中获取更多经验和反馈,同时希望大家支持和关注我们的产品。
    原文链接:http://www.iyunv.com/daizhj/archive/2011/02/21/1959511.html

  作者: daizhj, 代震军
  
  
      Tags: discuz!nt,redis  
  

  

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-88260-1-1.html 上篇帖子: NoSQL初探之人人都爱Redis:(4)Redis主从复制架构初步探索 下篇帖子: NoSQL初探之人人都爱Redis:(2)Redis API与常用数据类型简介
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

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