PHP利用MySQL锁解决高并发

20 篇文章 0 订阅

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

-- 库存管理表
CREATE TABLE `t_storage` (
 `f_id` int(11) unsigned NOT NULL AUTO_INCREMENT,
 `f_num` int(11) DEFAULT NULL,
 PRIMARY KEY (`f_id`)
) ENGINE=InnoDB AUTO_INCREMENT=2 DEFAULT CHARSET=latin1;

-- 订单管理表
CREATE TABLE `t_order` (
 `f_id` int(11) unsigned NOT NULL AUTO_INCREMENT,
 `f_num` int(11) DEFAULT NULL,
 PRIMARY KEY (`f_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 f_num from t_storage where f_id=1 limit 1";
$res = $pdo->query($sql)->fetch();
$num = $res['f_num'];

if($num > 0){
    $sql ="insert into t_order VALUES (null, $num)";
    $orderId = $pdo->query($sql);
    if($orderId){
        $sql="update t_storage set f_num=f_num-1 WHERE f_id=1";
        $pdo->query($sql);
    }
}

    我们预置库存是十个,然后执行ab测试查看结果。得到了订单共有12个,而库存表的库存也减到了-2,这显然不符合实际逻辑的。 

-- 查看结果
mysql> select * from t_storage;
+----+--------+
| f_id | f_num |
+----+--------+
| 1 |   -2 |
+----+--------+
1 row in set (0.00 sec)

mysql> select * from t_order;
+----+--------+
| f_id | f_num |
+----+--------+
| 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)

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

// 修改代码如下
$pdo = new PDO('mysql:host=127.0.0.1;port=3306;dbname=test','root','123456');
$pdo->beginTransaction(); // 开启事务
$sql = "select f_num from t_storage where f_id=1 for UPDATE "; // 利用for update 开启行锁
$res = $pdo->query($sql)->fetch();
$num = $res['f_num'];
if($num > 0) {
    $sql = "insert into t_order VALUES (null, $num)";
    $orderId = $pdo->query($sql);
    if($orderId) {
        $sql="update t_storage set f_num=f_num-1 WHERE f_id=1";
        if($pdo->query($sql)){
            $pdo->commit(); // 提交事务
        }else{
            $pdo->rollBack(); // 回滚
        }
    }else{
        $pdo->rollBack(); // 回滚
    }
}
-- 查看结果
mysql> select * from t_storage;
+----+--------+
| f_id | f_num |
+----+--------+
| 1 |   0 |
+----+--------+
1 row in set (0.00 sec)

mysql> select * from t_order;
+----+--------+
| f_id | f_num |
+----+--------+
| 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锁之后,对库存进行了有效的控制,很好的解决了第一段代码里面,因为并发引起的一些逻辑性的问题。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值