PHP利用Mysql鎖解決高并發(fā)的方法
前面寫過利用文件鎖來處理高并發(fā)的問題的,現(xiàn)在我們說另外一個(gè)處理方式,利用Mysql的鎖來解決高并發(fā)的問題
先看沒有利用事務(wù)的時(shí)候并發(fā)的后果
創(chuàng)建庫存管理表
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
測試代碼
$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ù)置庫存是十個(gè),然后執(zhí)行ab測試查看結(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è),而庫存表的庫存也減到了-2,這顯然不符合實(shí)際邏輯的;
下面我們來看利用數(shù)據(jù)庫行鎖來解決這個(gè)問題
修改代碼如下
$pdo = new PDO('mysql:host=127.0.0.1;port=3306; dbname=test','root','123456');
$pdo->beginTransaction();//開啟事務(wù)
$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();//提交事務(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鎖之后,對庫存進(jìn)行了有效的控制,很好的解決了第一段代碼里面,因?yàn)椴l(fā)引起的一些邏輯性的問題
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- mysql行鎖(for update)解決高并發(fā)問題
- Spring?Boot實(shí)戰(zhàn)解決高并發(fā)數(shù)據(jù)入庫之?Redis?緩存+MySQL?批量入庫問題
- MySQL高并發(fā)生成唯一訂單號的方法實(shí)現(xiàn)
- MySQL 數(shù)據(jù)庫如何解決高并發(fā)問題
- PHP+MySQL高并發(fā)加鎖事務(wù)處理問題解決方法
- MySQL中實(shí)現(xiàn)高性能高并發(fā)計(jì)數(shù)器方案(例如文章點(diǎn)擊數(shù))
- MySQL數(shù)據(jù)庫:?高并發(fā)電商場景下的架構(gòu)設(shè)計(jì)與優(yōu)化
相關(guān)文章
仿Aspnetpager的一個(gè)PHP分頁類代碼 附源碼下載
最近正在學(xué)習(xí)php,想拿一些demo來練習(xí)一下,想起前段時(shí)間自己用.net寫的仿Aspnetpager控件的分頁類,就琢磨著用PHP來實(shí)現(xiàn)仿aspnetpager的分頁功能,隨便復(fù)習(xí)下2012-10-10
laravel實(shí)現(xiàn)一個(gè)上傳圖片的接口,并建立軟鏈接,訪問圖片的方法
今天小編就為大家分享一篇laravel實(shí)現(xiàn)一個(gè)上傳圖片的接口,并建立軟鏈接,訪問圖片的方法,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2019-10-10
使用gd庫實(shí)現(xiàn)php服務(wù)端圖片裁剪和生成縮略圖功能分享
一般用戶上傳頭像時(shí),都會讓用戶自行裁剪圖片。那么php怎么實(shí)現(xiàn)這個(gè)功能呢?php中裁剪圖片主要使用gd庫的imagecopyresampled方法2013-12-12

