文档章节

cache缓存失效高并发读数据库的问题

昨日今日明日
 昨日今日明日
发布于 2012/12/11 00:19
字数 911
阅读 459
收藏 9

在网上看到有用其他语言实现的例子,这里用php写下,以加深理解。

用php具体代码实现

首先设计一个访问数据库的模型类

class model
{
    function getlist($catid)
    {
        
    }
}

方法一:

$cache = new cache(); //实例化一个已封装过的memcache类

function getnewlist($catid)
{
global $cache;

$key = "newslist-catid=".$catid; //设置一个cache key

$value = $cache->get($key); //获取key对应的cache

if($value == null) //key 对应的cache过期
{
    //按说过期的时候就应读取数据库 用getlist($catid)方法了。在小流量小并发时不会出现什么问题,当大并发时,可能会有几千个人同时访问到了cache失效,如果都去读取db的话,会造成db资源浪费甚至崩溃
   并发读取数据库势必会造成很慢的情况,如果还没有从数据库中读取出来并写入到cache中。这时候再有人访问过来,就又进入读取db的操作了。
    
    //因此增加锁
    $mutex_key = $key."-mutex"; //设置mutex key,用于支持本key的一个锁
    if($cache->add($mutex_key, 1) == true ) //对mutex_key 设置一个整数为1 的缓存,不计算马上存储。add方法就是第一次存储时返回true,如果已经存在返回false。这就相当于一个锁,避免了再次被执行
    {
$model = new model(); //实例化模型
          $value = $model->getlist($catid); //读取数据库获得catid的列表信息,数据库设置中应为catid建立索引
 $cache->set($key, $value);//刷新缓存
$cache->delete($mutex_key);//释放锁缓存
    }
    else//如果有并发访问过来,等待处理
    {
     sleep(2); //设置2秒的锁等待
                        getnewlist($catid);//尝试去读取结果
    }
}
         
        return $value;
}

方法二:

$cache = new cache(); //实例化一个memcache类

    function getnewlist($catid)
{
        global $cache;
        $expiretime = 360; //过期时间6分钟
        $key = 'newlist-catid='.$catid;
        $value = $cache->get($key);
        
        if($value == null) //缓存失效
        {
                //设置一个锁等待
                $mutex_key = $key.'-mutex';
                if($cache->add($mutex_key, 1) == true) //设置一个锁
                {
                        $model = new model(); //实例化模型
              $value = $model->getlist($catid); //读取数据库获得catid的列表信息,数据库设置中应为catid建立索引
    $cache->set($key, array('timeout' => time() + $expiretime, 'value' => $value), $expiretime*2);//刷新缓存,这步可以在cache类中封装

//这里有个问题,就是可能每个cache服务器会出现时间不一致,造成不是你预想的时间。对timeout赋值,如果只设置 $expiretime,如何获取时间节点问题。

    $cache->delete($mutex_key);//释放锁缓存
                }
                else
                {
                        sleep(2); //设置2秒的锁等待
                        getnewlist($catid);//尝试去读取结果
                }

        }
        else//如果还没有过期
{
//如果timeout已经过期,因为timeout设置的时间短,此时$key还没有到期,实际上是个伪过期时间点,timeout作为一个真正的到期时间点
if($value['timeout'] < time())
{
//设置锁
$mutex_key = $key.'-mutex';
if($cache->add($mutex_key, 1) == true)//首先捕获到并设置个锁
{
$value['timeout'] = time() +  $expiretime ; //马上延长过期时间
$cache->set($key, $value, $expiretime*2); //延长时间并保存,防止在高并发状态下,数据库读取时间长,而缓存已失效。


$model = new model(); //实例化模型
 $value = $model->getlist($catid); //读取数据库获得catid的列表信息,数据库设置中应为catid建立索引
 $cache->set($key, array('timeout' => time() + $expiretime, 'value' => $value), $expiretime*2);//刷新缓存,这步可以在cache类中封装


$cache->delete($mutex_key);//释放锁缓存

}
}
return $value['value'];
}
}

相比于方法一, 优点:避免cache失效时刻,大量请求无法执行add(key_mutex)而进入sleep锁定等待 缺点:代码复杂性增加,前台数据要求不那么严格时可以使用这里输入代码

© 著作权归作者所有

共有 人打赏支持
昨日今日明日
粉丝 3
博文 17
码字总数 7394
作品 0
程序员
私信 提问
加载中

评论(1)

kslr
kslr
$value = $$model->getlist($catid); 错误
缓存更新的套路

看到好些人在写更新缓存数据代码时,先删除缓存,然后再更新数据库,而后续的操作会把数据再装载的缓存中。然而,这个是逻辑是错误的。试想,两个并发操作,一个是更新操作,另一个是查询操作...

OSC一霸
2016/09/01
258
0
搞懂分布式技术15:缓存更新的套路

缓存更新的套路 看到好些人在写更新缓存数据代码时,先删除缓存,然后再更新数据库,而后续的操作会把数据再装载的缓存中。然而,这个是逻辑是错误的。试想,两个并发操作,一个是更新操作,...

你的猫大哥
06/23
0
0
怎样使用主流缓存更新策略来减少性能消耗?

在互联网项目开发中,缓存的应用是非常普遍了,缓存可以帮助页面提高加载速度,减少服务器或数据源的负载。 一般在项目中,最消耗性能的地方就是后端服务的数据库了。而数据库的读写频率常常...

IVAN-jsjwk
09/25
0
0
Java编程详细解析—淘宝大秒杀系统是如何设计的?

摘要 最初的秒杀系统的原型是淘宝详情上的定时上架功能,由于有些卖家为了吸引眼球,把价格压得很低。但这给的详情系统带来了很大压力,为了将这种突发流量隔离,才设计了秒杀系统,文章主要...

小刀爱编程
10/08
0
7
ASP.NET Core 中的缓存

目录 缓存的基本概念 缓存原理 缓存设计 分布式缓存 Memcache 与 Redis 的比较 缓存穿透,缓存击穿,缓存雪崩解决方案 数据一致性 使用内置 MemoryCache 使用分布式缓存 Redis 使用 Stackexc...

RoyZShare
08/14
0
0

没有更多内容

加载失败,请刷新页面

加载更多

小程序异步操作 跨js执行 在微信小程序里面实现跨页面通信

我们知道,在小程序里面一个页面的变化,是通过调用 setData 函数来实现的。所以想做到在二级页面里让一级页面产生变化,最 Quick And Dirty 的做法就是把一级页面的 this 传入到二级页面去,...

xiaogg
10分钟前
0
0
授于管理员登录其它用户

1.沙盒中,授予管理员登录 安全性控制==>登录访问权限政策

在山的那边
12分钟前
1
0
线程安全的CopyOnWriteArrayList介绍

证明CopyOnWriteArrayList是线程安全的 先写一段代码证明CopyOnWriteArrayList确实是线程安全的。 ReadThread.java import java.util.List; public class ReadThread implements Runnable {......

绝地逢生
15分钟前
0
0
Java重写的7个规则

几年前你可能会遇到这样一个面试题:“重写和重载的区别”、而现在随着科技的更迭、面试的问题越来越高级、面试官的问题也越来越深入、此文是上述面试题的一个延伸、让你从简单的重写规则中更...

architect刘源源
15分钟前
1
0
JavaScript异步编程:Generator与Async

从Promise开始,JavaScript就在引入新功能,来帮助更简单的方法来处理异步编程,帮助我们远离回调地狱。 Promise是下边要讲的Generator/yield与async/await的基础,希望你已经提前了解了它。...

前端攻城老湿
15分钟前
8
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部