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 相关文章推荐
file_get_contents获取不到网页内容的解决方法
Mar 07 PHP
php通过ksort()函数给关联数组按照键排序的方法
Mar 18 PHP
四个PHP非常实用的功能
Sep 29 PHP
PHP模板引擎Smarty内建函数foreach,foreachelse用法分析
Apr 11 PHP
php实现的一段简单概率相关代码
May 30 PHP
PHP根据树的前序遍历和中序遍历构造树并输出后序遍历的方法
Nov 10 PHP
PHP后期静态绑定实例浅析
Dec 21 PHP
PHP的mysqli_stat()函数讲解
Jan 23 PHP
Yii 使用intervention/image拓展实现图像处理功能
Jun 22 PHP
php实现文件上传基本验证
Mar 04 PHP
PHP+MySql实现一个简单的留言板
Jul 19 PHP
数据结构之利用PHP实现二分搜索树
Oct 25 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
Banner程序
2006/10/09 PHP
php的POSIX 函数以及进程测试的深入分析
2013/06/03 PHP
PHP APC的安装与使用详解
2013/06/13 PHP
使用PHP备份MySQL和网站发送到邮箱实例代码
2013/11/28 PHP
Zend Framework教程之Zend_Db_Table用法详解
2016/03/21 PHP
windows环境下使用Composer安装ThinkPHP5
2018/05/18 PHP
模仿jQuery each函数的链式调用
2009/07/22 Javascript
toString()一个会自动调用的方法
2010/02/08 Javascript
jQuery学习笔记之Helloworld
2010/12/22 Javascript
jquery select(列表)的操作(取值/赋值)
2011/03/16 Javascript
JQuery中阻止事件冒泡几种方式及其区别介绍
2014/01/15 Javascript
使用jsonp完美解决跨域问题
2014/11/27 Javascript
20分钟成功编写bootstrap响应式页面 就这么简单
2016/05/12 Javascript
微信小程序 教程之列表渲染
2016/10/18 Javascript
Easyui笔记2:实现datagrid多行删除的示例代码
2017/01/14 Javascript
mpvue写一个CPASS小程序的示例
2018/09/04 Javascript
Javascript实现秒表倒计时功能
2018/11/17 Javascript
javascript设计模式 ? 组合模式原理与应用实例分析
2020/04/14 Javascript
[06:07]刀塔密之二:攻之吾命受之吾幸
2014/07/03 DOTA
[05:31]干嘛呢兄弟!DOTA2 TI9语音轮盘部分出处
2019/05/14 DOTA
python matplotlib画图实例代码分享
2017/12/27 Python
Python编程argparse入门浅析
2018/02/07 Python
Python 仅获取响应头, 不获取实体的实例
2019/08/21 Python
详解Python的爬虫框架 Scrapy
2020/08/03 Python
使用Html5中的cavas画一面国旗
2019/09/25 HTML / CSS
捷克家电和家具购物网站:OKAY.cz
2020/07/23 全球购物
Java程序员面试90题
2013/10/19 面试题
运动会邀请函范文
2014/01/31 职场文书
高中生物教学反思
2014/02/05 职场文书
综合素质评价个性发展自我评价
2015/03/06 职场文书
单位提档介绍信
2015/10/22 职场文书
css背景和边框标签实例详解
2021/05/21 HTML / CSS
MySQL系列之十一 日志记录
2021/07/02 MySQL
HTML基础详解(上)
2021/10/16 HTML / CSS
Python Pygame实战之塔防游戏的实现
2022/03/17 Python
vue/cli 配置动态代理无需重启服务的方法
2022/05/20 Vue.js