PHP开发中解决并发问题的几种实现方法分析


Posted in PHP onNovember 13, 2017

本文实例讲述了PHP开发中解决并发问题的几种实现方法。分享给大家供大家参考,具体如下:

对于商品抢购等并发场景下,可能会出现超卖的现象,这时就需要解决并发所带来的这些问题了

在PHP语言中并没有原生的提供并发的解决方案,因此就需要借助其他方式来实现并发控制。

方案一:使用文件锁排它锁

flock函数用于获取文件的锁,这个锁同时只能被一个线程获取到,其它没有获取到锁的线程要么阻塞,要么获取失败

在获取到锁的时候,先查询库存,如果库存大于0,则进行下订单操作,减库存,然后释放锁

方案二:使用Mysql数据库提供的悲观锁

Innodb存储引擎支持行级锁,当某行数据被锁定时,其他进程不能对这行数据进行操作

先查询并锁定行:

select stock_num from table where id=1 for update
if(stock_num > 0){
//下订单
update table set stock_num=stock-1 where id=1
}

方案三:使用队列

将用户的下单请求依次存入一个队列中,后台用一个单独的进程处理队列中的下单请求

方案四:使用Redis

redis的操作都是原子性的,可以将商品的库存存入redis中,下单之前对库存进行decr操作,如果返回的值大于等于0等可以下单,否则不能下单,这种方式效率较高

if(redis->get('stock_num') > 0){
 stock_num = redis->decr('stock_num')
 if(stock_num >= 0){
 //下订单
 }else{
 //库存不足
 }
}else{
//库存不足
}

其他并发问题:

在现实应用中,很多情况下会把数据存入缓存,当缓存失效时,去数据库取数据并重新设置缓存,如果这时并发量很大,会有很多进程同时去数据库取数据,导致很多请求

穿透到数据库,而使数据库奔溃,这里可用文件锁来解决

$data = $cache->get('key');
if(!$data){
  $fp = fopen('lockfile');
  if(flock($fp, LOCK_EX)){
    $data = $cache->get('key');//拿到锁后再次检查缓存,这时可能已经有了
    if(!$data){
      $data = mysql->query();
      $cache->set('key', $data);
    }
    flock($fp, LOCK_UN);
  }
  fclose($fp);
}

说白了,要解决并发问题就必须要加锁,各种方案的本质都是加锁

希望本文所述对大家PHP程序设计有所帮助。

PHP 相关文章推荐
利用static实现表格的颜色隔行显示
Oct 09 PHP
PHP和Mysqlweb应用开发核心技术 第1部分 Php基础-3 代码组织和重用2
Jul 03 PHP
ThinkPHP 连接Oracle数据库的详细教程[全]
Jul 16 PHP
php验证是否是md5编码的简单代码
Apr 01 PHP
codeigniter中测试通过的分页类示例
Apr 17 PHP
PHP+Mysql+jQuery文件下载次数统计实例讲解
Oct 10 PHP
php的闭包(Closure)匿名函数初探
Feb 14 PHP
CodeIgniter开发实现支付宝接口调用的方法示例
Nov 14 PHP
php判断是否连接上网络的方法实例详解
Dec 14 PHP
用PHP的socket实现客户端到服务端的通信实例详解
Feb 04 PHP
PHP获取当前URL路径的处理方法(适用于多条件筛选列表)
Feb 10 PHP
自制PHP框架之路由与控制器
May 07 PHP
三个思路解决laravel上传文件报错:413 Request Entity Too Large问题
Nov 13 #PHP
kindeditor 加入七牛云上传的实例讲解
Nov 12 #PHP
Thinkphp5 微信公众号token验证不成功的原因及解决方法
Nov 12 #PHP
PHP 断点续传实例详解
Nov 11 #PHP
PHP+AJAX 投票器功能
Nov 11 #PHP
PHP实现双链表删除与插入节点的方法示例
Nov 11 #PHP
PHP实现基于栈的后缀表达式求值功能
Nov 10 #PHP
You might like
php中的观察者模式
2010/03/24 PHP
PHP 面向对象 final类与final方法
2010/05/05 PHP
无需重新编译php加入ftp扩展的解决方法
2013/02/07 PHP
在PHP中设置、使用、删除Cookie的解决方法
2013/05/06 PHP
php中apc缓存使用示例
2013/12/25 PHP
ThinkPHP3.1数据CURD操作快速入门
2014/06/19 PHP
php实现文本数据导入SQL SERVER
2015/05/17 PHP
PHP基于PDO扩展操作mysql数据库示例
2018/12/24 PHP
PHP的mysqli_stat()函数讲解
2019/01/23 PHP
详解new function(){}和function(){}() 区别分析
2008/03/22 Javascript
JavaScript 动态生成方法的例子
2009/07/22 Javascript
javascript dom追加内容实现示例
2013/09/21 Javascript
jQuery构造函数init参数分析续
2015/05/13 Javascript
JS+CSS实现的经典圆角下拉菜单效果代码
2015/10/21 Javascript
JavaScript性能优化之小知识总结
2015/11/20 Javascript
JS实现弹出居中的模式窗口示例
2016/06/20 Javascript
jQuery实现可展开折叠的导航效果示例
2016/09/12 Javascript
轻松理解Javascript变量的相关问题
2017/01/20 Javascript
vue.js 使用v-if v-else发现没有执行解决办法
2017/05/15 Javascript
JavaScript之Canvas_动力节点Java学院整理
2017/07/04 Javascript
详解node Async/Await 更好的异步编程解决方案
2018/05/10 Javascript
Vue中axios拦截器如何单独配置token
2019/12/27 Javascript
vue-quill-editor 自定义工具栏和自定义图片上传路径操作
2020/08/03 Javascript
微信小程序实现锚点跳转
2020/11/23 Javascript
[03:02]辉夜杯主赛事第二日 每日之星
2015/12/27 DOTA
[58:25]VP vs RNG 2019国际邀请赛小组赛 BO2 第一场 8.15
2019/08/17 DOTA
在Python的Tornado框架中实现简单的在线代理的教程
2015/05/02 Python
Python中文竖排显示的方法
2015/07/28 Python
利用python批量检查网站的可用性
2016/09/09 Python
python 容器总结整理
2017/04/04 Python
Python文件和流(实例讲解)
2017/09/12 Python
Python实现字典排序、按照list中字典的某个key排序的方法示例
2018/12/18 Python
Python面向对象之Web静态服务器
2019/09/03 Python
Python常用库大全及简要说明
2020/01/17 Python
安装不同版本的tensorflow与models方法实现
2021/02/20 Python
工艺工程师工作职责
2013/11/23 职场文书