原文链接
: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. WCF、WebAPI、WCFREST、WebService之间的区别总结(实用)

    在.net平台下,有大量的技术让你创建一个HTTP服务,像Web Service,WCF,现在又出了Web API.在.net平台下,你有很多的选择来构建一个HTTP Services.我分享一下我对 ...

  2. jQuery.Form.js 异步提交表单使用总结

    jQuery.Form.js 是一个用于使用jQuery异步提交表单的插件,它使用方法简单,支持同步和异步两种方式提交. 第一步:引入jQuery与jQuery.Form.js <script ...

  3. [日常] Go语言圣经--包和文件-导入包习题

    1.每个包都有一个全局唯一的导入路径 2.按照惯例,一个包的名字和包的导入路径的最后一个字段相同 练习 2.2: 写一个通用的单位转换程序,用类似cf程序的方式从命令行读取参数,如果缺省的话则是从标准 ...

  4. webhttpbinding、basichttpbinding和wshttpbinding的区别

    webhttpbinding是REST风格的绑定,您只需点击一个URL,然后从Web服务中获取大量XML或JSON. basichttpbinding和wshttpbinding是两个基于SOAP的绑 ...

  5. ubuntu 17.10 安装后的应用软件安装

    目录 安装 sogou 拼音 安装markdown编辑器 安装codeblocks 下载工具uGet+aira2 安装QT 安装remarkable(markdown工具) 安装StarUML(UML ...

  6. FZU1759(SummerTrainingDay04-B 欧拉降幂公式)

    Problem 1759 Super A^B mod C Accept: 1056    Submit: 3444Time Limit: 1000 mSec    Memory Limit : 327 ...

  7. Python标准库--UUID

    UUID(Universally Unique Identifier)是128位通用唯一识别码,通常由32字节的字符串表示.它可以保证时间和空间的唯一性,也称为GUID,全称为:UUID —— Uni ...

  8. 关于子元素的margin-top对父级容器无效

    如果不想看那么长,看下面这句话就好了. 刚开始我没看到这个总结时一直是使用自己摸索出来paddin-top解决,发现该方式并不好.亲测给父级加一个overflow不为visiable的属性就直接解决了 ...

  9. 说说对npm的开发模式和生产模式的理解

    nodejs这些年的发展非常快,相信没有哪个前端不知道的了,npm也成为了前端开发中经常用到了的一个命令.那么npm不是只用一个 "npm install xxx"命令就够了吗?实 ...

  10. h5新增加的存储方法

    h4中使用的cookie把用户信息保存在客户端浏览器,但是它受到很多限制. 大小:最多能存储4k 带宽:它是随着http请求一起发送到服务器的,因此浪费一部分的带宽. 复杂度:操作复杂. h5新增加了 ...