电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

您的位置:首頁技術文章
文章詳情頁

MySQL中SELECT+UPDATE處理并發更新問題解決方案

瀏覽:6日期:2023-10-17 08:07:13

這篇文章主要介紹了MySQL中SELECT+UPDATE處理并發更新問題解決方案分享,需要的朋友可以參考下。

問題背景

假設MySQL數據庫有一張會員表vip_member(InnoDB表),結構如下:

MySQL中SELECT+UPDATE處理并發更新問題解決方案

當一個會員想續買會員(只能續買1個月、3個月或6個月)時,必須滿足以下業務要求:

如果end_at早于當前時間,則設置start_at為當前時間,end_at為當前時間加上續買的月數如果end_at等于或晚于當前時間,則設置end_at=end_at+續買的月數續買后active_status必須為1(即被激活)問題分析

對于上面這種情況,我們一般會先SELECT查出這條記錄,然后根據查出記錄的end_at再UPDATE start_at和end_at,偽代碼如下(為uid是1001的會員續1個月):

vipMember = SELECT * FROM vip_member WHERE uid=1001 LIMIT 1 # 查uid為1001的會員 if vipMember.end_at < NOW(): UPDATE vip_member SET start_at=NOW(), end_at=DATE_ADD(NOW(), INTERVAL 1 MONTH), active_status=1, updated_at=NOW() WHERE uid=1001 else: UPDATE vip_member SET end_at=DATE_ADD(end_at, INTERVAL 1 MONTH), active_status=1, updated_at=NOW() WHERE uid=1001

假如同時有兩個線程執行上面的代碼,很顯然存在“數據覆蓋”問題(即一個是續1個月,一個續2個月,但最終可能只續了2個月,而不是加起來的3個月)。

解決方案

A、我想到的第一種方案是把SELECT和UPDATE合成一條SQL

如下:

UPDATE vip_member SET start_at = CASE WHEN end_at < NOW() THEN NOW()ELSE start_atEND, end_at = CASEWHEN end_at < NOW()THEN DATE_ADD(NOW(), INTERVAL #duration:INTEGER# MONTH)ELSE DATE_ADD(end_at, INTERVAL #duration:INTEGER# MONTH)END, active_status=1, updated_at=NOW()WHERE uid=#uid:BIGINT#LIMIT 1;

B、第二種方案:事務,即用一個事務來包裹上面的SELECT+UPDATE操作

那么是否包上事務就萬事大吉了呢?

顯然不是。因為如果同時有兩個事務都分別SELECT到相同的vip_member記錄,那么一樣的會發生數據覆蓋問題。那有什么辦法可以解決呢?難道要設置事務隔離級別為SERIALIZABLE,考慮到性能不現實。

我們知道InnoDB支持行鎖。查看MySQL官方文檔(innodb locking reads)了解到InnoDB在讀取行數據時可以加兩種鎖:讀共享鎖和寫獨占鎖。

讀共享鎖是通過下面這樣的SQL獲得的:

SELECT * FROM parent WHERE NAME = ’Jones’ LOCK IN SHARE MODE;

如果事務A獲得了先獲得了讀共享鎖,那么事務B之后仍然可以讀取加了讀共享鎖的行數據,但必須等事務A commit或者roll back之后才可以更新或者刪除加了讀共享鎖的行數據。

SELECT counter_field FROM child_codes FOR UPDATE;UPDATE child_codes SET counter_field = counter_field + 1;

如果事務A先獲得了某行的寫共享鎖,那么事務B就必須等待事務A commit或者roll back之后才可以訪問行數據。

顯然要解決會員狀態更新問題,不能加讀共享鎖,只能加寫共享鎖,即將前面的SQL改寫成如下:

vipMember = SELECT * FROM vip_member WHERE uid=1001 LIMIT 1 FOR UPDATE # 查uid為1001的會員if vipMember.end_at < NOW(): UPDATE vip_member SET start_at=NOW(), end_at=DATE_ADD(NOW(), INTERVAL 1 MONTH), active_status=1, updated_at=NOW() WHERE uid=1001else: UPDATE vip_member SET end_at=DATE_ADD(end_at, INTERVAL 1 MONTH), active_status=1, updated_at=NOW() WHERE uid=1001

另外這里特別提醒下:UPDATE/DELETE SQL盡量帶上WHERE條件并在WHERE條件中設定索引過濾條件,否則會鎖表,性能可想而知有多差了。

C、第三種方案:樂觀鎖,類CAS機制

第二種加鎖方案是一種悲觀鎖機制。而且SELECT...FOR UPDATE方式也不太常用,聯想到CAS實現的樂觀鎖機制,于是我想到了第三種解決方案:樂觀鎖。

具體來說也挺簡單,首先SELECT SQL不作任何修改,然后在UPDATE SQL的WHERE條件中加上SELECT出來的vip_memer的end_at條件。如下:

vipMember = SELECT * FROM vip_member WHERE uid=1001 LIMIT 1 # 查uid為1001的會員cur_end_at = vipMember.end_atif vipMember.end_at < NOW(): UPDATE vip_member SET start_at=NOW(), end_at=DATE_ADD(NOW(), INTERVAL 1 MONTH), active_status=1, updated_at=NOW() WHERE uid=1001 AND end_at=cur_end_atelse: UPDATE vip_member SET end_at=DATE_ADD(end_at, INTERVAL 1 MONTH), active_status=1, updated_at=NOW() WHERE uid=1001 AND end_at=cur_end_at

這樣可以根據UPDATE返回值來判斷是否更新成功,如果返回值是0則表明存在并發更新,那么只需要重試一下就好了。

方案比較

三種方案各自優劣也許眾說紛紜,只說說我自己的看法:

第一種方案利用一條比較復雜的SQL解決問題,不利于維護,因為把具體業務糅在SQL里了,以后修改業務時不但需要讀懂這條SQL,還很有可能會修改成更復雜的SQL第二種方案寫獨占鎖,可以解決問題,但不常用第三種方案應該是比較中庸的解決方案,并且甚至可以不加事務,也是我個人推薦的方案

此外,樂觀鎖和悲觀鎖的選擇一般是這樣的(參考了文末第二篇資料):

如果對讀的響應度要求非常高,比如證券交易系統,那么適合用樂觀鎖,因為悲觀鎖會阻塞讀如果讀遠多于寫,那么也適合用樂觀鎖,因為用悲觀鎖會導致大量讀被少量的寫阻塞如果寫操作頻繁并且沖突比例很高,那么適合用悲觀寫獨占鎖
標簽: MySQL 數據庫
相關文章:
主站蜘蛛池模板: 工程管道/塑料管材/pvc排水管/ppr给水管/pe双壁波纹管等品牌管材批发厂家-河南洁尔康建材 | 步入式高低温测试箱|海向仪器| 上海办公室装修,办公楼装修设计,办公空间设计,企业展厅设计_写艺装饰公司 | 复合土工膜厂家|hdpe防渗土工膜|复合防渗土工布|玻璃纤维|双向塑料土工格栅-安徽路建新材料有限公司 | 【电子厂招聘_普工招工网_工厂招聘信息平台】-工立方打工网 | 找果网 | 苹果手机找回方法,苹果iPhone手机丢了找回,认准找果网! | 艾默生变频器,艾默生ct,变频器,ct驱动器,广州艾默生变频器,供水专用变频器,风机变频器,电梯变频器,艾默生变频器代理-广州市盟雄贸易有限公司官方网站-艾默生变频器应用解决方案服务商 | 碳刷_刷握_集电环_恒压簧_电刷厂家-上海丹臻机电科技有限公司 | 烘箱-工业烘箱-工业电炉-实验室干燥箱 - 苏州华洁烘箱制造有限公司 | 消电检公司,消电检价格,北京消电检报告-北京设施检测公司-亿杰(北京)消防工程有限公司 | 99文库_实习生实用的范文资料文库站 | 天坛家具官网 | 电动打包机_气动打包机_钢带捆扎机_废纸打包机_手动捆扎机 | 针焰试验仪,灼热丝试验仪,漏电起痕试验仪,水平垂直燃烧试验仪 - 苏州亚诺天下仪器有限公司 | 双相钢_双相不锈钢_双相钢圆钢棒_双相不锈钢报价「海新双相钢」 双能x射线骨密度检测仪_dxa骨密度仪_双能x线骨密度仪_品牌厂家【品源医疗】 | 披萨石_披萨盘_电器家电隔热绵加工定制_佛山市南海区西樵南方综合保温材料厂 | 机床导轨_导轨板_滚轮导轨-上海旻佑精密机械有限公司 | 透平油真空滤油机-变压器油板框滤油机-滤油车-华之源过滤设备 | 钢绞线万能材料试验机-全自动恒应力两用机-混凝土恒应力压力试验机-北京科达京威科技发展有限公司 | 宝元数控系统|对刀仪厂家|东莞机器人控制系统|东莞安川伺服-【鑫天驰智能科技】 | 卓能JOINTLEAN端子连接器厂家-专业提供PCB接线端子|轨道式端子|重载连接器|欧式连接器等电气连接产品和服务 | 自进式锚杆-自钻式中空注浆锚杆-洛阳恒诺锚固锚杆生产厂家 | 工业硝酸钠,硝酸钠厂家-淄博「文海工贸」 | LHH药品稳定性试验箱-BPS系列恒温恒湿箱-意大利超低温冰箱-上海一恒科学仪器有限公司 | 小程序开发公司_APP开发多少钱_软件开发定制_微信小程序制作_客户销售管理软件-济南小溪畅流网络科技有限公司 | 世纪豪门官网 世纪豪门集成吊顶加盟电话 世纪豪门售后电话 | ORP控制器_ORP电极价格-上优泰百科 | 不锈钢丸厂家,铝丸,铸钢丸-淄博智源铸造材料有限公司 | 湖州织里童装_女童男童中大童装_款式多尺码全_织里儿童网【官网】-嘉兴嘉乐网络科技有限公司 | 仿古瓦,仿古金属瓦,铝瓦,铜瓦,铝合金瓦-西安东申景观艺术工程有限公司 | 悬浮拼装地板_幼儿园_篮球场_悬浮拼接地板-山东悬浮拼装地板厂家 | 便携式XPDM露点仪-在线式防爆露点仪-增强型烟气分析仪-约克仪器 冰雕-冰雪世界-大型冰雕展制作公司-赛北冰雕官网 | 深圳APP开发公司_软件APP定制开发/外包制作-红匣子科技 | 高速龙门架厂家_监控杆_多功能灯杆_信号灯杆_锂电池太阳能路灯-鑫世源照明 | 电脑刺绣_绣花厂家_绣花章仔_织唛厂家-[源欣刺绣]潮牌刺绣打版定制绣花加工厂家 | 泉州陶瓷pc砖_园林景观砖厂家_石英砖地铺石价格 _福建暴风石英砖 | 滁州高低温冲击试验箱厂家_安徽高低温试验箱价格|安徽希尔伯特 | 企小优-企业数字化转型服务商_网络推广_网络推广公司 | 物流之家新闻网-最新物流新闻|物流资讯|物流政策|物流网-匡匡奈斯物流科技 | 威廉希尔WilliamHill·足球(中国)体育官方网站 | 阿里巴巴诚信通温州、台州、宁波、嘉兴授权渠道商-浙江联欣科技提供阿里会员办理 |