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 31 PHP
PHP防止跨域提交表单
Nov 01 PHP
彻底删除thinkphp3.1案例blog标签的方法
Dec 05 PHP
php实现两个数组相加的方法
Feb 17 PHP
PHP 9 大缓存技术总结
Sep 17 PHP
PHP代码判断设备是手机还是平板电脑(两种方法)
Oct 19 PHP
详解php中反射的应用
Mar 15 PHP
PHP实现的随机IP函数【国内IP段】
Jul 20 PHP
PHP+redis实现的悲观锁机制示例
Jun 12 PHP
PHP实现一个限制实例化次数的类示例
Sep 16 PHP
Laravel 前端资源配置教程
Oct 18 PHP
PHP利用curl发送HTTP请求的实例代码
Jul 09 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
php mssql 数据库分页SQL语句
2008/12/16 PHP
PHP 截取字符串专题集合
2010/08/19 PHP
php 在字符串指定位置插入新字符的简单实现
2016/06/28 PHP
php简单构造json多维数组的方法示例
2017/06/08 PHP
PHP判断访客是否手机端(移动端浏览器)访问的方法总结【4种方法】
2019/03/27 PHP
悬浮数字的实现案例
2014/02/19 Javascript
JS通过ajax动态读取xml文件内容的方法
2015/03/24 Javascript
详解ECMAScript6入门--Class对象
2017/04/27 Javascript
lhgcalendar时间插件限制只能选择三个月的实现方法
2017/07/03 Javascript
vue子父组件通信的实现代码
2017/07/09 Javascript
nodejs简单实现TCP服务器端和客户端的聊天功能示例
2018/01/04 NodeJs
AngularJS修改model值时,显示内容不变的实例
2018/09/13 Javascript
如何使用JavaScript实现栈与队列
2019/06/24 Javascript
easyUI 实现的后台分页与前台显示功能示例
2020/06/01 Javascript
Nuxt.js 静态资源和打包的操作
2020/11/06 Javascript
在GitHub Pages上使用Pelican搭建博客的教程
2015/04/25 Python
Python实现模拟登录及表单提交的方法
2015/07/25 Python
Python实现简单的四则运算计算器
2016/11/02 Python
python opencv3实现人脸识别(windows)
2018/05/25 Python
pycharm使用matplotlib.pyplot不显示图形的解决方法
2018/10/28 Python
django写用户登录判定并跳转制定页面的实例
2019/08/21 Python
IE矩阵Matrix滤镜旋转与缩放及如何结合transform
2012/11/29 HTML / CSS
使用CSS3的::selection改变选中文本颜色的方法
2015/09/29 HTML / CSS
瑜伽服装品牌:露露柠檬(lululemon athletica)
2017/06/04 全球购物
Carter’s官方旗舰店:美国受欢迎的婴童服装品牌
2018/01/21 全球购物
10条PHP编程习惯
2014/05/26 面试题
保险公司晨会主持词
2014/03/22 职场文书
《春天来了》教学反思
2014/04/07 职场文书
记账会计岗位职责
2014/06/16 职场文书
学习雷锋精神活动总结
2015/02/06 职场文书
2015年安全生产工作总结范文
2015/04/02 职场文书
医药公司开票员岗位职责
2015/04/15 职场文书
律政俏佳人观后感
2015/06/09 职场文书
2016年大学生社区服务活动总结
2016/04/06 职场文书
导游词之香港-太平山顶
2019/10/18 职场文书
python中Pyqt5使用Qlabel标签播放视频
2022/04/22 Python