原文链接
:https://blog.csdn.net/tim_phper/article/details/54949404

概述:

项目当中经常要考虑数据高并发的情况,为了避免并发导致出现一些资源重复请求的问题,可以使用缓存加锁机制。 
例如取微信access_token不加锁可能会导致非常严重的后果。

准备:

缓存锁,顾名思义,当然离不开缓存,这篇文章用到的redis缓存,可以根据自己的需要要选择合适缓存。 
缓存锁的原理是在进行操作A之前,先在缓存中存放一个唯一的key,然后就进行对应操作A,而如果同时有其他一样的操作B并发时,因为操作B也需要存放key才能进行操作,而key是唯一的,所以操作B是无法进行存放一样key,也就是说操作B还没开始就被中断了。而当操作A完成之后,或者报错之后就进行释放锁,也就是从缓存当中删除对应的key。除此之外,当操作A操作时间过长时,我们也应该释放锁。具体流程如下图: 

分析:

我这里用的是CI的框架,并不是原生的,但CI的框架也很容易看懂。

//缓存锁class
class Locker { private $_CI; public function __construct() {
$this->_CI = & get_instance();
} /**
* 取锁
* @param type $key
* @param type $timeout 默认锁能锁10秒,10秒后锁自动解除
* @return type
*/
public function lock($key, $timeout = 10) {
$now = time();
$cache_key = $this->_get_key($key); // 如果redis中没有$cache_key,则加锁成功
if ($this->_CI->cache->setnx($cache_key, $now) === true) {
return true;
} // 如果加锁时间超过最大锁时间,则自动解锁,并将锁赋予新的进程
// 特别注意:并发情况下,使用getset方法可以保证只有一个进程获取到锁
// 只有当$last_set_time == $_last_set_time时才是保证当前进程获取到锁
$last_set_time = $this->_CI->cache->get($cache_key);
if ($now - $last_set_time > $timeout) {
$_last_set_time = $this->_CI->cache->getset($cache_key, $now);
if ($last_set_time == $_last_set_time) {
return true;
}
} return false;
} /**
* 释放锁
* @param string $key
* @return boolean
*/
public function release($key) {
$this->_CI->cache->delete($this->_get_key($key));
} private $_lock_key_prefix = 'lock_';
private function _get_key($key) {
return $this->_lock_key_prefix . $key;
} }
  //缓存锁利用过程
////////////////////////////// 加锁进行拿奖 开始 加锁 ////////////////////////
$lock_key = 'get_prize_lock_' . $user->id; //创建唯一key
if ( ! $this->locker->lock($lock_key)) { //判断缓存中是否已经存在唯一key
return $this->send_json(false, '系统繁忙,请稍后重试...','',2);
} // 检查是否已领取过
$where = array(
'act_uuid' => $act_uuid,
'openid' => $user->wx_user_open_id,
'share_from_result_id < ' => 1,
);
$count = $this->data_result_model->count_by($where);
if ( $count > 0 ){
$this->locker->release($lock_key); //每次操作出错都释放锁
return $this->send_json(true, '您已领取过卡券礼包了噢.');
}
//进行数据操作
foreach ($coupon_pkg as $code => $coupon) {
$dateTime = date('Y-m-d H:i:s');
$record = array(
'user_id' => $user->user_id,
'act_uuid' => $act_uuid, ); $this->eggs_data_result_model->add($record);
}
////////////////////////////// 加锁进行拿奖 结束 释放锁 ////////////////////////
$this->locker->release($lock_key); //操作完成释放锁 return $this->send_json(true, '卡券礼包已成功发放。',$act_uuid); //操作完成

总结:

PHP处理数据时,并发情况经常出现,缓存锁机制真的是好处理方法,不过值得注意是经常检查缓存的运行状况,因为一旦缓存挂了,那么整个系统都会出错,无法正常运行的。

PHP缓存锁原理及利用的更多相关文章

  1. 利用多写Redis实现分布式锁原理与实现分析(转)

    利用多写Redis实现分布式锁原理与实现分析   一.关于分布式锁 关于分布式锁,可能绝大部分人都会或多或少涉及到. 我举二个例子:场景一:从前端界面发起一笔支付请求,如果前端没有做防重处理,那么可能 ...

  2. Java进阶专题(二十五) 分布式锁原理与实现

    前言 ​ 现如今很多系统都会基于分布式或微服务思想完成对系统的架构设计.那么在这一个系统中,就会存在若干个微服务,而且服务间也会产生相互通信调用.那么既然产生了服务调用,就必然会存在服务调用延迟或失败 ...

  3. zookeeper 分布式锁原理

    zookeeper 分布式锁原理: 1 大家也许都很熟悉了多个线程或者多个进程间的共享锁的实现方式了,但是在分布式场景中我们会面临多个Server之间的锁的问题,实现的复杂度比较高.利用基于googl ...

  4. MySql 缓存查询原理与缓存监控 和 索引监控

    MySql缓存查询原理与缓存监控 And 索引监控 by:授客 QQ:1033553122 查询缓存 1.查询缓存操作原理 mysql执行查询语句之前,把查询语句同查询缓存中的语句进行比较,且是按字节 ...

  5. MIP缓存加速原理 MIP不仅仅只是CDN

    什么是MIP?我想我们现在都知道.可是你真的了解MIP吗?MIP加速原理是什么?MIP 是用 CDN 做加速的么?准确答案是:是,但不只是. 很多人并认为MIP百度排名会靠前,甚至权重会提高?作为一个 ...

  6. Redis分布式锁原理

    1. Redis分布式锁原理 1.1. Redisson 现在最流行的redis分布式锁就是Redisson了,来看看它的底层原理就了解redis是如何使用分布式锁的了 1.2. 原理分析 分布式锁要 ...

  7. AQS学习(二) AQS互斥模式与ReenterLock可重入锁原理解析

    1. MyAQS介绍    在这个系列博客中,我们会参考着jdk的AbstractQueuedLongSynchronizer,从零开始自己动手实现一个AQS(MyAQS).通过模仿,自己造轮子来学习 ...

  8. Redisson 实现分布式锁原理分析

    Redisson 实现分布式锁原理分析   写在前面 在了解分布式锁具体实现方案之前,我们应该先思考一下使用分布式锁必须要考虑的一些问题.​ 互斥性:在任意时刻,只能有一个进程持有锁. 防死锁:即使有 ...

  9. 【Android - 进阶】之图片三级缓存的原理及实现

    在Android开发中,如果图片过多,而我们又没有对图片进行有效的缓存,就很容易导致OOM(Out Of Memory)错误.因此,图片的缓存是非常重要的,尤其是对图片非常多的应用.现在很多框架都做了 ...

随机推荐

  1. SQL Server T—SQL 表连接

    一  笛卡尔积 select  *  from  表1,表2 将两表的记录遍历显示 二表的横向连接 1   使用外键关系作为条件 select  *  from   表1,表2  where   表1 ...

  2. C# 利用反射将枚举绑定到下拉框

    前言:反射(Reflection)是.NET提供给开发者的一个强大工具,尽管作为.NET框架的使用者,很多时候不会用到反射.但在一些情况下,尤其是在开发一些基础框架或公共类库时,使用反射会使系统架构更 ...

  3. 36.Linux驱动调试-根据oops定位错误代码行

    1.当驱动有误时,比如,访问的内存地址是非法的,便会打印一大串的oops出来 1.1以LED驱动为例 将open()函数里的ioremap()屏蔽掉,直接使用物理地址的GPIOF,如下图所示: 1.2 ...

  4. Android-事件分发机制框架概述

    http://www.jianshu.com/p/e99b5e8bd67b http://blog.csdn.net/guolin_blog/article/details/9097463 https ...

  5. HDU4960(SummerTrainingDay03-F dp)

    Another OCD Patient Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 131072/131072 K (Java/Ot ...

  6. 初学HTML-3

    标题标签:<h#>...</h#>,从h1到h6,字号由大变小. 段落标签:<p>...</p>,在浏览器中独占一行. 空格:" " ...

  7. vue实现倒计时的插件 时间戳 刷新 跳转 都不影响

    工作当中需要开发一个倒计时插件,于是开始网上先拿来主义,发现好多倒计时的插件,刷新都会变成从头再来,于是自己用vue2.0写了一个插件,测试已经通过,直接上代码 如下是组件代码: <templa ...

  8. 【代码笔记】iOS-屏幕根据键盘自动的变化高度

    一,效果图. 二,代码. ViewController.h #import <UIKit/UIKit.h> @interface ViewController : UIViewContro ...

  9. 在Silverlight中动态绑定页面报表(PageReport)的数据源

    ActiveReports 7中引入了一种新的报表模型——PageReport(页面布局报表),这种报表模型又细分了两种具体显示形式: o    固定页面布局报表模型(FPL)是ActiveRepor ...

  10. Jenkins 为Jenkins添加Windows Slave远程执行python项目脚本

    为Jenkins添加Windows Slave远程执行python项目脚本   by:授客 QQ:1033553122 测试环境 JAVA JDK 1.7.0_13 (jdk-7u13-windows ...