国产三级农村妇女在线,国产精品毛片a∨一区二区三区,国产乱子伦视频大全,国产精品色拉拉,国产欧美日韩一区二区三区,

首頁 > 技術(shù) > IOS開發(fā)

thinkphp6使用mysql悲觀鎖解決商品超賣問題的實現(xiàn)

IOS開發(fā) 2023-02-16 22:22:00

悲觀鎖介紹(百科):

悲觀鎖,正如其名,它指的是對數(shù)據(jù)被外界(包括本系統(tǒng)當(dāng)前的其他事務(wù),以及來自外部系統(tǒng)的事務(wù)處理)修改持保守態(tài)度,因此,在整個數(shù)據(jù)處理過程中,將數(shù)據(jù)處于鎖定狀態(tài)。悲觀鎖的實現(xiàn),往往依靠數(shù)據(jù)庫提供的鎖機(jī)制(也只有數(shù)據(jù)庫層提供的鎖機(jī)制才能真正保證數(shù)據(jù)訪問的排他性,否則,即使在本系統(tǒng)中實現(xiàn)了加鎖機(jī)制,也無法保證外部系統(tǒng)不會修改數(shù)據(jù))。

使用場景舉例:以MySQL InnoDB為例

商品goods表,假設(shè)商品的id為1,購買數(shù)量為1,status為1表示上架中,2表示下架?,F(xiàn)在用戶購買此商品,在不是高并發(fā)的情況下處理邏輯是:

  • 查找此商品的信息;
  • 檢查商品庫存是否大于購買數(shù)量;
  • 修改商品庫存和銷量;

上面這種場景在高并發(fā)訪問的情況下很可能會出現(xiàn)問題。如果商品庫存是100個,高并發(fā)的情況下可能會有1000個同時訪問,在到達(dá)第2步的時候,都會檢測通過。這樣會出現(xiàn)商品庫存是-900個的情況。顯然著不滿足需求?。?!

商品表結(jié)構(gòu):

CREATE TABLE `goods` (
  `id` int(11) unsigned NOT NULL AUTO_INCREMENT,
  `name` varchar(100) COLLATE utf8_unicode_ci NOT NULL DEFAULT '',
  `status` tinyint(1) NOT NULL DEFAULT '1',
  `total` int(11) NOT NULL DEFAULT '0',
  `sell` int(11) NOT NULL DEFAULT '100',
  `price` decimal(10,2) NOT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8 COLLATE=utf8_unicode_ci;
 
INSERT INTO `test`.`goods`(`id`, `name`, `status`, `total`, `sell`, `price`) VALUES (1, '商品', 1, 0, 100, 15.00);

訂單表結(jié)構(gòu):

CREATE TABLE `orders` (
  `id` int(11) unsigned NOT NULL AUTO_INCREMENT,
  `uid` int(11) NOT NULL DEFAULT '0',
  `create_time` datetime NOT NULL,
  `status` tinyint(1) NOT NULL DEFAULT '1',
  `goods_id` int(11) NOT NULL DEFAULT '0',
  `order_no` varchar(200) COLLATE utf8_unicode_ci NOT NULL DEFAULT '',
  PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8 COLLATE=utf8_unicode_ci;

使用悲觀鎖處理。

當(dāng)我們在查詢出goods信息后就把當(dāng)前的數(shù)據(jù)鎖定,直到我們修改完畢后再解鎖。那么在這個過程中,因為goods被鎖定了,就不會出現(xiàn)有第三者來對其進(jìn)行修改了。

注:要使用悲觀鎖,我們必須關(guān)閉mysql數(shù)據(jù)庫的自動提交屬性,因為MySQL默認(rèn)使用autocommit模式,也就是說,當(dāng)你執(zhí)行一個更新操作后,MySQL會立刻將結(jié)果進(jìn)行提交。Thinkphp6中使用事務(wù),手動進(jìn)行提交回滾。

<?php
 
namespace appcontroller;
 
use appBaseController;
use thinkfacadeDb;
 
class Test extends BaseController
{
 
    /**
     * 不加鎖
     * @return string|void
     */
    public function test_1()
    {
        $num = 1;
        $goods_id = 1;
        Db::startTrans();
        try {
            $where = [];
            $where['id'] = $goods_id;
            $where['status'] = 1;
            $goods_info = Db::table('goods')->where($where)->find();
            if (empty($goods_info)) {
                return '商品不存在';
            }
            $total = $goods_info['total'];
            $sell = $goods_info['sell'];
            if ($total < $num) {
                return '庫存不足';
            }
            $data['total'] = $total-$num;
            $data['sell'] = $sell+$num;
            $res = Db::table('goods')->where(['id'=>$goods_id])->update($data);
            $order_data = [];
            $order_data['uid'] = rand(1000,9999);
            $order_data['status'] = 1;
            $order_data['create_time'] = date('Y-m-d H:i:s');
            $order_data['goods_id'] = $goods_id;
            $order_data['order_no'] = date('YmdHis').rand(1000,10000);
            $order_res = Db::table('orders')->insert($order_data);
            Db::commit();
        } catch (Exception $e) {
            // 回滾事務(wù)
            Db::rollback();
            echo $e->getMessage();
            exit('rollback');
        }
        echo '請求成功';
    }
 
    /**
     * 加鎖--悲觀鎖
     * @return string|void
     */
    public function test_2()
    {
        $num = 1;
        $goods_id = 1;
        Db::startTrans();
        try {
            $where = [];
            $where['id'] = $goods_id;
            $where['status'] = 1;
            $goods_info = Db::table('goods')->lock(true)->where($where)->find();
            if (empty($goods_info)) {
                return '商品不存在';
            }
            $total = $goods_info['total'];
            $sell = $goods_info['sell'];
            if ($total < $num) {
                return '庫存不足';
            }
            $data['total'] = $total-$num;
            $data['sell'] = $sell+$num;
            $res = Db::table('goods')->where(['id'=>$goods_id])->update($data);
            $order_data = [];
            $order_data['uid'] = rand(1000,9999);
            $order_data['status'] = 1;
            $order_data['goods_id'] = $goods_id;
            $order_data['order_no'] = date('YmdHis').rand(1000,10000);
            $order_data['create_time'] = date('Y-m-d H:i:s');
            $order_res = Db::table('orders')->insert($order_data);
            Db::commit();
        } catch (Exception $e) {
            // 回滾事務(wù)
            Db::rollback();
            echo $e->getMessage();
            exit('rollback');
 
        }
        echo '請求成功';
    }
 
}

使用jmeter工具測試,創(chuàng)建線程測試組:

關(guān)于使用jmeter創(chuàng)建測試高并發(fā)例子,可查看:使用JMeter進(jìn)行高并發(fā)測試_左右..的博客-CSDN博客_jmeter高并發(fā)測試

這里模擬1s內(nèi)1000個用戶同時訪問

thinkphp6使用mysql悲觀鎖解決商品超賣問題的實現(xiàn)(圖1)

?創(chuàng)建http請求:

thinkphp6使用mysql悲觀鎖解決商品超賣問題的實現(xiàn)(圖2)

添加察看結(jié)果樹:

thinkphp6使用mysql悲觀鎖解決商品超賣問題的實現(xiàn)(圖3)

?測試開始:

100個商品不加鎖的結(jié)果:

thinkphp6使用mysql悲觀鎖解決商品超賣問題的實現(xiàn)(圖4)

100個商品庫存,生成訂單187個,超賣87個商品,這在項目開發(fā)中是絕對不允許的。

100個商品,加鎖結(jié)果:

thinkphp6使用mysql悲觀鎖解決商品超賣問題的實現(xiàn)(圖5)

?加鎖,得到解決。

到此這篇關(guān)于thinkphp6使用mysql悲觀鎖解決商品超賣問題的實現(xiàn)的文章就介紹到這了,更多相關(guān)thinkphp6 商品超賣內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

TAg

加載中~

本網(wǎng)站LOGO受版權(quán)及商標(biāo)保護(hù),版權(quán)登記號:國作登字-2022-F-10126915,未經(jīng)湖南木星科技官方許可,嚴(yán)禁使用。
Copyright ? 2012-2022 湖南木星科技有限公司(木星網(wǎng))版權(quán)所有
轉(zhuǎn)載內(nèi)容版權(quán)歸作者及來源網(wǎng)站所有,本站原創(chuàng)內(nèi)容轉(zhuǎn)載請注明來源,商業(yè)媒體及紙媒請先聯(lián)系:aishangyiwan@126.com

工信部備案號:湘ICP備19012813號-5