mysql如何才能保證數(shù)據(jù)的一致性
mysql如何保證數(shù)據(jù)一致性
MySQL保證數(shù)據(jù)一致性的主要方式有以下幾種:
- 事務(wù)支持:MySQL 支持事務(wù),通過事務(wù)可以保證數(shù)據(jù)庫(kù)操作要么全部執(zhí)行成功,要么全部失敗回滾,從而確保數(shù)據(jù)的一致性。
- ACID特性:MySQL 遵循 ACID(原子性、一致性、隔離性、持久性)特性,其中一致性是指事務(wù)執(zhí)行前后數(shù)據(jù)的狀態(tài)保持一致。在事務(wù)中,所有的修改必須符合相關(guān)約束,這樣才能保證一致性。
- 唯一約束和外鍵約束:MySQL 可以通過設(shè)置唯一約束和外鍵約束來(lái)保證數(shù)據(jù)一致性。唯一約束保證某列或者幾列的取值都是唯一的,外鍵約束可以保證參照完整性,確保關(guān)聯(lián)表之間的數(shù)據(jù)一致性。
- 觸發(fā)器:MySQL 中的觸發(fā)器(Trigger)可以在指定的條件下自動(dòng)執(zhí)行特定的操作,通過觸發(fā)器可以實(shí)現(xiàn)數(shù)據(jù)一致性的操作,比如在插入、更新、刪除數(shù)據(jù)時(shí)做相應(yīng)的檢查和處理。
- 存儲(chǔ)過程:MySQL 中的存儲(chǔ)過程(Stored Procedure)可以封裝一系列的 SQL 語(yǔ)句,可以在存儲(chǔ)過程中實(shí)現(xiàn)一些復(fù)雜的業(yè)務(wù)邏輯,確保數(shù)據(jù)操作的一致性。
綜上所述,MySQL 可以通過事務(wù)支持、ACID特性、約束、觸發(fā)器和存儲(chǔ)過程等多種手段來(lái)保證數(shù)據(jù)的一致性。
redis和mysql如何保持?jǐn)?shù)據(jù)一致性
在高并發(fā)的場(chǎng)景下,大量的請(qǐng)求直接訪問Mysql很容易造成性能問題。
通常情況,我們都會(huì)用Redis來(lái)做數(shù)據(jù)的緩存,削減對(duì)數(shù)據(jù)庫(kù)的請(qǐng)求。
但是,Mysql和Redis是兩種不同的數(shù)據(jù)庫(kù),如何保證兩者之間的數(shù)據(jù)一致性呢?
導(dǎo)致數(shù)據(jù)不一致的原因
1、在高并發(fā)的業(yè)務(wù)場(chǎng)景下,數(shù)據(jù)庫(kù)大多數(shù)情況都是用戶并發(fā)訪問最薄弱的環(huán)節(jié)。
2、所以,就需要使用redis做一個(gè)緩沖操作,讓請(qǐng)求先訪問到redis,而不是直接訪問MySQL等數(shù)據(jù)庫(kù)。
3、讀取緩存步驟一般沒有什么問題,但是一旦涉及到數(shù)據(jù)更新:數(shù)據(jù)庫(kù)和緩存更新,就容易出現(xiàn)緩存(Redis)和數(shù)據(jù)庫(kù)(MySQL)間的數(shù)據(jù)一致性問題。
4、這個(gè)業(yè)務(wù)場(chǎng)景,主要是解決讀數(shù)據(jù)從Redis緩存,一般都是按照下圖的流程來(lái)進(jìn)行業(yè)務(wù)操作。
分析
1、先刪除redis中緩存
(1)如果先刪除Redis緩存數(shù)據(jù),然而還沒有來(lái)得及寫入MySQL,另一個(gè)線程就來(lái)讀取。
(2)這個(gè)時(shí)候發(fā)現(xiàn)緩存為空,則去Mysql數(shù)據(jù)庫(kù)中讀取舊數(shù)據(jù)寫入緩存,此時(shí)緩存中為臟數(shù)據(jù)。
(3)然后數(shù)據(jù)庫(kù)更新后發(fā)現(xiàn)Redis和Mysql出現(xiàn)了數(shù)據(jù)不一致的問題。
2、后刪除redis中緩存
(1)如果先寫了庫(kù),然后再刪除緩存,不幸的寫庫(kù)的線程掛了,導(dǎo)致了緩存沒有刪除
(2)這個(gè)時(shí)候就會(huì)直接讀取舊緩存,最終也導(dǎo)致了數(shù)據(jù)不一致情況
(3)因?yàn)閷懞妥x是并發(fā)的,沒法保證順序,就會(huì)出現(xiàn)緩存和數(shù)據(jù)庫(kù)的數(shù)據(jù)不一致的問題
實(shí)現(xiàn)方案
Redis中的對(duì)key刪除操作要橫跨整個(gè)高并發(fā)時(shí)間周期,無(wú)論是主動(dòng)對(duì)key的刪除還是對(duì)key設(shè)置的過期時(shí)間,都需要后延至高并發(fā)場(chǎng)景過后的普通負(fù)載環(huán)節(jié)中。
這樣可以最大限度的實(shí)現(xiàn)redis和MySQL中的數(shù)據(jù)一致性。
且高并發(fā)場(chǎng)景頻繁對(duì)redis中的key進(jìn)行刪除操作,本身就會(huì)降低數(shù)據(jù)的安全性,同時(shí)降低程序反應(yīng)時(shí)間,降低用戶體驗(yàn)。
方案一: 延時(shí)雙刪策略
在寫庫(kù)前后都進(jìn)行redis.del(key)操作,并且設(shè)定合理的超時(shí)時(shí)間。
具體步驟是:
1)先刪除緩存
2)再寫數(shù)據(jù)庫(kù)
3)休眠500毫秒(根據(jù)具體的業(yè)務(wù)時(shí)間來(lái)定)
4)再次刪除緩存。
那么,這個(gè)500毫秒怎么確定的,具體該休眠多久呢?
需要評(píng)估自己的項(xiàng)目的讀數(shù)據(jù)業(yè)務(wù)邏輯的耗時(shí)。這么做的目的,就是確保讀請(qǐng)求結(jié)束,寫請(qǐng)求可以刪除讀請(qǐng)求造成的緩存臟數(shù)據(jù)。
當(dāng)然,這種策略還要考慮 redis 和數(shù)據(jù)庫(kù)主從同步的耗時(shí)。最后的寫數(shù)據(jù)的休眠時(shí)間:則在讀數(shù)據(jù)業(yè)務(wù)邏輯的耗時(shí)的基礎(chǔ)上,加上幾百ms即可。比如:休眠1秒。
方案二:合理設(shè)置緩存的過期時(shí)間
從理論上來(lái)說,給緩存設(shè)置過期時(shí)間,是保證最終一致性的解決方案。所有的寫操作以數(shù)據(jù)庫(kù)為準(zhǔn),只要到達(dá)緩存過期時(shí)間,則后面的讀請(qǐng)求自然會(huì)從數(shù)據(jù)庫(kù)中讀取新值然后回填緩存。
總結(jié)
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
Mysql插入數(shù)據(jù)方式(insert into 、replace into解析)
這篇文章主要介紹了Mysql插入數(shù)據(jù)方式(insert into 、replace into解析),具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-01-01新裝MySql后登錄出現(xiàn)root帳號(hào)提示mysql ERROR 1045 (28000): Access denied
這篇文章主要介紹了新裝MySql后登錄出現(xiàn)root帳號(hào)提示mysql ERROR 1045 (28000): Access denied for use的解決辦法,需要的朋友可以參考下2017-01-01MySQL如何統(tǒng)計(jì)一個(gè)數(shù)據(jù)庫(kù)所有表的數(shù)據(jù)量
最近在做統(tǒng)計(jì)想查找一個(gè)數(shù)據(jù)庫(kù)里基本所有的表數(shù)據(jù)量,下面這篇文章主要給大家介紹了關(guān)于MySQL如何統(tǒng)計(jì)一個(gè)數(shù)據(jù)庫(kù)所有表的數(shù)據(jù)量的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),需要的朋友可以參考下2022-04-04超簡(jiǎn)單的qps統(tǒng)計(jì)方法(推薦)
下面小編就為大家?guī)?lái)一篇超簡(jiǎn)單的qps統(tǒng)計(jì)方法(推薦)。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來(lái)看看吧2017-03-03安裝MySQL 5.7出現(xiàn)報(bào)錯(cuò):unknown variable ‘mysqlx_port
這篇文章主要介紹了安裝MySQL 5.7出現(xiàn)報(bào)錯(cuò):unknown variable ‘mysqlx_port=0.0‘的解決方法,文中通過圖文結(jié)合的方式介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作有一定的幫助,需要的朋友可以參考下2024-06-06