PHP利用Mysql锁解决高并发的方法


Posted in PHP onSeptember 04, 2018

前面写过利用文件锁来处理高并发的问题的,现在我们说另外一个处理方式,利用Mysql的锁来解决高并发的问题

先看没有利用事务的时候并发的后果

创建库存管理表

CREATE TABLE `storage` (
 `id` int(11) unsigned NOT NULL AUTO_INCREMENT,
 `number` int(11) DEFAULT NULL,
 PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=2 DEFAULT CHARSET=latin1

创建订单管理表

CREATE TABLE `order` (
 `id` int(11) unsigned NOT NULL AUTO_INCREMENT,
 `number` int(11) DEFAULT NULL,
 PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=34 DEFAULT CHARSET=latin1

测试代码

$pdo = new PDO('mysql:host=127.0.0.1;port=3306; dbname=test','root','123456');
$sql="select `number` from storage where id=1 limit 1";
$res = $pdo->query($sql)->fetch();
$number = $res['number'];

if($number>0)
{
  $sql ="insert into `order` VALUES (null,$number)";
  
  $order_id = $pdo->query($sql);
  if($order_id)
  {

    $sql="update storage set `number`=`number`-1 WHERE id=1";
    $pdo->query($sql);
  }
}

我们预置库存是十个,然后执行ab测试查看结果

mysql> select * from storage
  -> ;
+----+--------+
| id | number |
+----+--------+
| 1 |   -2 |
+----+--------+
1 row in set (0.00 sec)

mysql> select * from `order`;
+----+--------+
| id | number |
+----+--------+
| 22 |   10 |
| 23 |   10 |
| 24 |   8 |
| 25 |   8 |
| 26 |   7 |
| 27 |   6 |
| 28 |   4 |
| 29 |   3 |
| 30 |   2 |
| 31 |   2 |
| 32 |   2 |
| 33 |   1 |
+----+--------+
12 rows in set (0.00 sec)

得到了订单共有12个,而库存表的库存也减到了-2,这显然不符合实际逻辑的;

下面我们来看利用数据库行锁来解决这个问题

修改代码如下

$pdo = new PDO('mysql:host=127.0.0.1;port=3306; dbname=test','root','123456');
$pdo->beginTransaction();//开启事务
$sql="select `number` from storage where id=1 for UPDATE ";//利用for update 开启行锁
$res = $pdo->query($sql)->fetch();
$number = $res['number'];

if($number>0)
{
  $sql ="insert into `order` VALUES (null,$number)";

  $order_id = $pdo->query($sql);
  if($order_id)
  {

    $sql="update storage set `number`=`number`-1 WHERE id=1";
    if($pdo->query($sql))
    {
      $pdo->commit();//提交事务
    }
    else
    {
      $pdo->rollBack();//回滚
    }

  }
  else
  {
    $pdo->rollBack();//回滚
  }
}

查看结果

mysql> select * from storage;
+----+--------+
| id | number |
+----+--------+
| 1 |   0 |
+----+------
--+
1 row in set (0.00 sec)

mysql> select * from `order`;
+----+--------+
| id | number |
+----+--------+
| 1 |   10 |
| 2 |   9 |
| 3 |   8 |
| 4 |   7 |
| 5 |   6 |
| 6 |   5 |
| 7 |   4 |
| 8 |   3 |
| 9 |   2 |
| 10 |   1 |
+----+--------+
10 rows in set (0.00 sec)

很明显在利用了mysql锁之后,对库存进行了有效的控制,很好的解决了第一段代码里面,因为并发引起的一些逻辑性的问题

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

PHP 相关文章推荐
国内php原创论坛
Oct 09 PHP
PHP调用MySQL的存储过程的实现代码
Aug 12 PHP
一贴学会PHP 新手入门教程
Aug 03 PHP
细谈php中SQL注入攻击与XSS攻击
Jun 10 PHP
解析php dirname()与__FILE__常量的应用
Jun 24 PHP
PHP输出当前进程所有变量/常量/模块/函数/类的示例
Nov 07 PHP
php实现mysql封装类示例
May 07 PHP
php通过正则表达式记取数据来读取xml的方法
Mar 09 PHP
php实现可运算的验证码
Nov 10 PHP
php使用curl通过代理获取数据的实现方法
May 16 PHP
浅谈ThinkPHP5.0版本和ThinkPHP3.2版本的区别
Jun 17 PHP
Laravel5.0+框架邮件发送功能实现方法图文与实例详解
Apr 23 PHP
TP5(thinkPHP5)框架基于ajax与后台数据交互操作简单示例
Sep 03 #PHP
PHP实现的AES双向加密解密功能示例【128位】
Sep 03 #PHP
PHP实现的日历功能示例
Sep 01 #PHP
thinkPHP5框架分页样式类完整示例
Sep 01 #PHP
php操作mongodb封装类与用法实例
Sep 01 #PHP
thinkphp集成前端脚手架Vue-cli的教程图解
Aug 30 #PHP
Django中通过定时任务触发页面静态化的处理方式
Aug 29 #PHP
You might like
实例讲解yii2.0在php命令行中运行的步骤
2015/12/01 PHP
浅谈PHP的数据库接口和技术
2016/12/09 PHP
Laravel关系模型指定条件查询方法
2019/10/10 PHP
总结AJAX相关JS代码片段和浏览器模型
2007/08/15 Javascript
过虑特殊字符输入的js代码
2010/08/05 Javascript
表单元素的submit()方法和onsubmit事件应用概述
2013/02/01 Javascript
通过$(this)使用jQuery包装后的方法或属性
2014/05/18 Javascript
基于jquery实现的文字向上跑动类似跑马灯的效果
2014/06/17 Javascript
基于jquery实现页面滚动时顶部导航显示隐藏
2020/04/20 Javascript
Bootstrap实现默认导航栏效果
2020/09/21 Javascript
安装使用Mongoose配合Node.js操作MongoDB的基础教程
2016/03/01 Javascript
jquery实现图片上传前本地预览功能
2016/05/10 Javascript
超链接怎么正确调用javascript函数
2016/05/23 Javascript
微信小程序 页面之间传参实例详解
2017/01/13 Javascript
JS轮播图中缓动函数的封装
2020/11/25 Javascript
Angular中ng-options下拉数据默认值的设定方法
2017/06/21 Javascript
node文件批量重命名的方法示例
2017/10/23 Javascript
vue里面使用mui的弹出日期选择插件实例
2018/09/16 Javascript
通过实例解析JavaScript常用排序算法
2020/09/02 Javascript
[54:10]Spirit vs NB Supermajor小组赛 A组败者组决赛 BO3 第一场 6.2
2018/06/03 DOTA
Python入门篇之列表和元组
2014/10/17 Python
一篇文章搞懂Python的类与对象名称空间
2018/12/10 Python
Python多进程写入同一文件的方法
2019/01/14 Python
python实现年会抽奖程序
2019/01/22 Python
Python 使用PyQt5 完成选择文件或目录的对话框方法
2019/06/27 Python
新建文件时Pycharm中自动设置头部模板信息的方法
2020/04/17 Python
TensorFlow实现批量归一化操作的示例
2020/04/22 Python
pandas数据分组groupby()和统计函数agg()的使用
2021/03/04 Python
CSS3实现渐变背景兼容问题
2020/05/06 HTML / CSS
英国网络托管和域名领导者:Web Hosting UK
2017/10/15 全球购物
荷兰时尚精品店:Labels Fashion
2020/03/22 全球购物
拾金不昧表扬信范文
2014/01/11 职场文书
2014年节能减排工作总结
2014/12/06 职场文书
2015年安全保卫工作总结
2015/05/14 职场文书
步步惊心观后感
2015/06/12 职场文书
Meta增速拉垮,元宇宙难当重任
2022/04/29 数码科技