這篇文章將為大家詳細(xì)講解有關(guān)如何解決PHP+MySQL高并發(fā)加鎖事務(wù)處理的問題,小編覺得挺實用的,因此分享給大家做個參考,希望大家閱讀完這篇文章后可以有所收獲。
為準(zhǔn)格爾等地區(qū)用戶提供了全套網(wǎng)頁設(shè)計制作服務(wù),及準(zhǔn)格爾網(wǎng)站建設(shè)行業(yè)解決方案。主營業(yè)務(wù)為網(wǎng)站制作、成都網(wǎng)站制作、準(zhǔn)格爾網(wǎng)站設(shè)計,以傳統(tǒng)方式定制建設(shè)網(wǎng)站,并提供域名空間備案等一條龍服務(wù),秉承以專業(yè)、用心的態(tài)度為用戶提供真誠的服務(wù)。我們深信只要達(dá)到每一位用戶的要求,就會得到認(rèn)可,從而選擇與我們長期合作。這樣,我們也可以走得更遠(yuǎn)!具體如下:
1、背景:
現(xiàn)在有這樣的需求,插入數(shù)據(jù)時,判斷test表有無username為‘mraz'的數(shù)據(jù),無則插入,有則提示“已插入”,目的就是想只插入一條username為‘mraz'的記錄。
2、一般程序邏輯如下:
$conn = mysqli_connect('127.0.0.1', 'root', '111111') or die(mysqli_error()); mysqli_select_db($conn, 'mraz'); $rs = mysqli_query($conn, 'SELECT count(*) as total FROM test WHERE username = "mraz" '); $row = mysqli_fetch_array($rs); if($row['total']>0){ exit('exist'); } mysqli_query($conn, "insert into test(username) values ('mraz')"); var_dump('error:'.mysqli_errno($conn)); $insert_id = mysqli_insert_id($conn); echo 'insert_id:'.$insert_id.'
'; mysqli_free_result($rs); mysqli_close($conn);
3、一般少量請求的時候,程序邏輯不會有問題。但是一旦高并發(fā)請求執(zhí)行的話,程序并沒有按預(yù)期執(zhí)行,會插入多條username為‘mraz'的記錄。
4、解決方案:利用mysql的FOR UPDATE 語句和事務(wù)的隔離性。注意的是FOR UPDATE僅適用于InnoDB,且必須在事務(wù)(BEGIN/COMMIT)中才能生效。
調(diào)整代碼后如下:
$conn = mysqli_connect('127.0.0.1', 'root', '111111') or die(mysqli_error()); mysqli_select_db($conn, 'mraz'); mysqli_query($conn, 'BEGIN'); $rs = mysqli_query($conn, 'SELECT count(*) as total FROM test WHERE username = "mraz" FOR UPDATE'); $row = mysqli_fetch_array($rs); if($row['total']>0){ exit('exist'); } mysqli_query($conn, "insert into test(username) values ('mraz')"); var_dump('error:'.mysqli_errno($conn)); $insert_id = mysqli_insert_id($conn); mysqli_query($conn, 'COMMIT'); echo 'insert_id:'.$insert_id.'
'; mysqli_free_result($rs); mysqli_close($conn);
5、再利用php的curl模擬高并發(fā)請求該php腳本,查看數(shù)據(jù)庫會只有一條username為‘mraz'的記錄。達(dá)到程序執(zhí)行的預(yù)期結(jié)果~
關(guān)于“如何解決PHP+MySQL高并發(fā)加鎖事務(wù)處理的問題”這篇文章就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,使各位可以學(xué)到更多知識,如果覺得文章不錯,請把它分享出去讓更多的人看到。