关于Memcache mutex设计模式的.net实现

之前在网上看过memcache-mutex的场景分析和实现代码,这里将.net方式加以实现,当然这里主要是依据原文的伪代码照猫画虎,以此做为总结及记录。如果您对相应实现感兴趣可以尝试使用本文提供的代码进行测试,如果有问题请及时与我联系。

原文链接:http://timyang.net/programming/memcache-mutex/

本地链接:http://www.cnblogs.com/daizhj/articles/1959704.html

为了实现原文中的对象到期时间属性,定义了一个基类,其信息如下:

[Serializable]
public class CacheObj
{
        /// <summary>
        /// 数据绝对到期时间,默认为当前时间开始三分钟后失效
        /// </summary>
        public DateTime ExpireTime = DateTime.Now.AddMinutes(3);

/// <summary>
        /// 数据相对有效时间,单位:秒。默认为30秒有效期
        /// </summary>
        public int TimeOut = 30;
}

这样所有要放到memcached的对象只要继承该对象就OK了,比如下面的用户信息类:

/// <summary>
/// 用户信息
/// </summary>
[Serializable]
public class UserInfo : CacheObj
{
        public string UserName;
        public int Age;
        public string Email;

public override string ToString()
        {
            return "UserName:" + UserName + "  Age:" + Age + "  Email:" + Email;
        }
}

下面是原文中方式一的实现代码:

MemcachedClient mc = MemCachedManager.CacheClient;
//方一
public UserInfo GetCacheData1(string key)
{
    UserInfo value = mc.Get(key) as UserInfo;
    if (value == null)
    {
        // 3 分钟到期.在delete操作执行之前,当前key_mutex add只能被添加一次并返回true
        if (mc.Add(key + "_mutex", key + "_mutex", DateTime.Now.AddMinutes(3)) == true)
        {
            value = new UserInfo() { UserName = "daizhj", Email = "[email protected]" };// db.get(key);//从加载数据
            mc.Set(key, value);
            mc.Delete(key + "_mutex");                    
        }
        else
        {
            System.Threading.Thread.Sleep(500);//如果设置过短,可能上面set语法还未生效
            value = mc.Get(key) as UserInfo;//sleep之后重试读取cache数据
        }
    }
    return value;
}

下面是方式2的代码:

//方法二
public UserInfo GetCacheData2(string key)
{
    UserInfo value = mc.Get(key) as UserInfo;
    if (value == null)
    {
        // 3 分钟到期,在delete之前,当前key_mutex add只能被添加一次并返回true
        if (mc.Add(key + "_mutex", "add_mutex", DateTime.Now.AddMinutes(3)) == true)
        {
            value = new UserInfo() { UserName = "daizhj", Email = "[email protected]" };// db.get(key);//从加载数据
            mc.Set(key, value);
            mc.Delete(key + "_mutex");
        }
        else
        {
            System.Threading.Thread.Sleep(500);//如果设置过短,可能上面set语法还未生效
            value = mc.Get(key) as UserInfo;//sleep之后重试读取cache数据
        }
    }
    else
    {
        if (value.ExpireTime <= DateTime.Now)
        {
            //有值但已过期 
            if (mc.Add(key + "_mutex", "add_mutex", DateTime.Now.AddMinutes(3)) == true)
            {
                value.ExpireTime = DateTime.Now.AddSeconds(value.TimeOut);
                //这只是为了让它先暂时有效(后面即将更新该过期数据),这样做主要防止避免cache失效时刻大量请求获取不到mutex并进行sleep,注意这里设置成有效会导致其它线程会暂时读到脏数据
                mc.Set(key, value, DateTime.Now.AddSeconds(value.TimeOut * 2));//这里*2是为了让memcached缓存数据更长时间,因为真正校验到期时间用ExpireTime来判断

//从数据源加载最新数据
                value = new UserInfo() { UserName = "daizhenjun", Email = "[email protected]" };// db.get(key);
                value.ExpireTime = DateTime.Now.AddSeconds(value.TimeOut);
                mc.Set(key, value, DateTime.Now.AddSeconds(value.TimeOut * 2));
                mc.Delete(key + "_mutex");
            }
            else
            {
                System.Threading.Thread.Sleep(500);//如果设置过短,可能上面set语法还未生效
                value = mc.Get(key) as UserInfo;//sleep之后重试读取cache数据
            }
        }
    }
    return value;
}

无论使用那种方式,都会带来代码复杂性增大(尤其第二种),另外还有就是与memcached额外的连接及存储开销(key_mutex本身存储也要消耗资源)。因为除非是高并发场景下同时更新memcached,否则这两种方式需要斟酌使用。

源码下载地址:/Files/daizhj/MemcachedApp.rar

【示例位于MemcachedApp\sample\MutexSample.aspx】

时间: 2024-10-21 13:22:57

关于Memcache mutex设计模式的.net实现的相关文章

Memcache的mutex设计模式 -- 高并发解决方案

场景 Mutex主要用于有大量并发访问并存在cache过期的场合,如 首页top 10, 由数据库加载到memcache缓存n分钟: 微博中名人的content cache, 一旦不存在会大量请求不能命中并加载数据库: 需要执行多个IO操作生成的数据存在cache中, 比如查询db多次: 问题 在大并发的场合,当cache失效时,大量并发同时取不到cache,会同一瞬间去访问db并回设cache,可能会给系统带来潜在的超负荷风险.我们曾经在线上系统出现过类似故障. 解决方法  方法一 高并发时,

Web开发基本准则-55实录-缓存策略

郑昀 创建于2013年2月 郑昀 最后更新于2013年10月26日 提纲: Web访问安全 缓存策略 存储介质连接池 业务降级 并发请求的处理 关键词: 会话串号,Cache-Control头域,缓存穿透,缓存集体失效,缓存重建,缓存雪崩,缓存永不过期,缓存计数器, 二,缓存策略 这里的“缓存”概念不只限于服务器端的“缓存”. 2.1.防会话串号 如果你收到一个投诉,说访问“我的个人中心”页面时进入其他人的帐号,至少订单列表上显示的不是自己的.此时,技术支持人员可以提三个问题,第一,对页面上显示

设计模式之PHP项目应用——单例模式设计Memcache和Redis操作类

1 单例模式简单介绍 单例模式是一种经常使用的软件设计模式. 在它的核心结构中仅仅包括一个被称为单例类的特殊类. 通过单例模式能够保证系统中一个类仅仅有一个实例并且该实例易于外界訪问.从而方便对实例个数的控制并节约系统资源.假设希望在系统中某个类的对象仅仅能存在一个.单例模式是最好的解决方式. 2 模式核心思想 1)某个类仅仅能有一个实例: 2)它必须自行创建这个实例: 3)它必须自行向整个系统提供这个实例. 3 模式架构图 4 项目应用 4.1 需求说明 CleverCode在实际的PHP项目

设计模式-单例模式(Go语言描述)

这篇博客我们继续来看设计模式,今天带来的是一个最简单而且最常用的模式-单例模式.那什么是单例模式呢?相信大家最它最熟悉不过了,那我们就来快速的了解一下它的定义. 保证一个类仅有一个实例,并提供一个访问它的全局访问点. 这个解释足够简单.说白了就是假如我们希望我们在我们的系统中该类仅仅存在1个或0个该类的实例.虽然单例模式很简单,但是熟悉java的同学可能了解,单例模式有很多写法,懒汉式.饿汉式.双重锁... 这么多形式,难道有什么目的?确实,不过他们的目的很明确,就是保证在一种特殊情况下的单例-

Memcache架构新思考

2011年初Marc Kwiatkowski通过[email protected]介绍了Facebook的Memcache架构,现在重新审视这个架构,仍有很多方面在业界保持先进性.作为weibo内部数据处理量最大,对数据延迟最敏感的部门,基于本厂2年多来对mc的使用心得,我在本文总结对MC架构的一些新思考. 1. Memcache使用中的雷区 通常你可能考虑不到,但又隐藏在某处等着你踩的称之为"雷". 带宽和连接数 Memcache具有很高吞吐能力,[email protected]中

设计模式学习第二天:设计模式总结——1单例模式

原文博主:http://blog.csdn.net/zhaoguiqun/article/details/6026763 一. 单例(Singleton)模式      单例模式的特点:• 单例类只能有一个实例. • 单例类必须自己创建自己的唯一实例. • 单例类必须给所有其它对象提供这一实例. 单例模式应用:每台计算机可以有若干个打印机,但只能有一个Printer Spooler,避免两个打印作业同时输出到打印机. 一个具有自动编号主键的表可以有多个用户同时使用,但数据库中只能有一个地方分配下

设计模式系列之单例模式

单例模式是使用最广泛,也最简单的设计模式之一,作用是保证一个类只有一个实例.单例模式是对全局变量的一种改进,避免全局变量污染命名空间.因为以下几个原因,全局变量不能作为单例的实现方式: 1. 不能保证只有一个全局变量 2. 静态初始化时可能没有足够的信息创建对象 3. c++中全局对象的构造顺序是未定义的,如果单件之间存在依赖将可能产生错误 单例模式的实现代码很简单: //singleton.hpp #ifndef SINGLETON_HPP #define SINGLETON_HPP clas

设计模式(1)单例模式

程序中有时候需要保存全局的数据,比如程序的配置文件,需要随时检索的.比如程序中有些变量需要全局保存全局用,这时候我们不想用一个全局变量来保存 这时候,可以使用单例模式,从名称可以看出,单例模式就是类的实例全局只创建一个.怎么样才能保存只创建一个实例呢? 我们可以设置标识位,创建过的就不再创建了.下面是单例的简单实现 ? 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33

设计模式之单例模式:singleton

1.设计模式是什么? 设计模式其实就是前人总结,代表了最佳实践,对于软件开发过程中对象的封装模式,也是各种复杂问题,极好解耦性的解决方案. ------------------------------------------------------------------------------------------- 下面我们来说一下单例模式的基本概念和代码: 单例类保证了全局只有唯一一个实例对象 单例提供获取这个唯一实例的接口 其实就是保证一个类中出现对象的全局唯一性. 首先对于单例模式而