佳木斯湛栽影视文化发展公司

主頁(yè) > 知識(shí)庫(kù) > PHP利用Mysql鎖解決高并發(fā)的方法

PHP利用Mysql鎖解決高并發(fā)的方法

熱門標(biāo)簽:阿里云 服務(wù)器配置 電子圍欄 Linux服務(wù)器 銀行業(yè)務(wù) 團(tuán)購(gòu)網(wǎng)站 科大訊飛語(yǔ)音識(shí)別系統(tǒng) Mysql連接數(shù)設(shè)置

前面寫(xiě)過(guò)利用文件鎖來(lái)處理高并發(fā)的問(wèn)題的,現(xiàn)在我們說(shuō)另外一個(gè)處理方式,利用Mysql的鎖來(lái)解決高并發(fā)的問(wèn)題

先看沒(méi)有利用事務(wù)的時(shí)候并發(fā)的后果

創(chuàng)建庫(kù)存管理表

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

創(chuàng)建訂單管理表

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

測(cè)試代碼

$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);
  }
}

我們預(yù)置庫(kù)存是十個(gè),然后執(zhí)行ab測(cè)試查看結(jié)果

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個(gè),而庫(kù)存表的庫(kù)存也減到了-2,這顯然不符合實(shí)際邏輯的;

下面我們來(lái)看利用數(shù)據(jù)庫(kù)行鎖來(lái)解決這個(gè)問(wèn)題

修改代碼如下

$pdo = new PDO('mysql:host=127.0.0.1;port=3306; dbname=test','root','123456');
$pdo->beginTransaction();//開(kāi)啟事務(wù)
$sql="select `number` from storage where id=1 for UPDATE ";//利用for update 開(kāi)啟行鎖
$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();//提交事務(wù)
    }
    else
    {
      $pdo->rollBack();//回滾
    }

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

查看結(jié)果

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鎖之后,對(duì)庫(kù)存進(jìn)行了有效的控制,很好的解決了第一段代碼里面,因?yàn)椴l(fā)引起的一些邏輯性的問(wèn)題

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

您可能感興趣的文章:
  • php多進(jìn)程并發(fā)編程防止出現(xiàn)僵尸進(jìn)程的方法分析
  • PHP高并發(fā)和大流量解決方案整理
  • PHP 并發(fā)場(chǎng)景的幾種解決方案
  • PHP下用Swoole實(shí)現(xiàn)Actor并發(fā)模型的方法
  • php多進(jìn)程模擬并發(fā)事務(wù)產(chǎn)生的問(wèn)題小結(jié)
  • PHP curl批處理及多請(qǐng)求并發(fā)實(shí)現(xiàn)方法分析
  • php curl批處理實(shí)現(xiàn)可控并發(fā)異步操作示例
  • PHP使用curl_multi實(shí)現(xiàn)并發(fā)請(qǐng)求的方法示例
  • 詳解PHP服務(wù)器如何在有限的資源里最大提升并發(fā)能力

標(biāo)簽:衢州 棗莊 大理 江蘇 蚌埠 廣元 萍鄉(xiāng) 衡水

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《PHP利用Mysql鎖解決高并發(fā)的方法》,本文關(guān)鍵詞  ;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問(wèn)題,煩請(qǐng)?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無(wú)關(guān)。
  • 相關(guān)文章
  • 收縮
    • 微信客服
    • 微信二維碼
    • 電話咨詢

    • 400-1100-266
    苏州市| 余江县| 鹰潭市| 民丰县| 乐东| 衡水市| 天津市| 阜新| 永福县| 泰州市| 长春市| 泽库县| 石屏县| 慈利县| 霸州市| 保山市| 砀山县| 镇远县| 宣城市| 晋州市| 泉州市| 会泽县| 葵青区| 新蔡县| 刚察县| 大港区| 泉州市| 南溪县| 博罗县| 拜城县| 启东市| 固安县| 乌兰察布市| 淮阳县| 建平县| 苏尼特右旗| 闽侯县| 若羌县| 江永县| 牟定县| 双辽市|