亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

一文詳解MYSQL的多版本并發(fā)控制MVCC(Multi-Version Concurrency Control)

 更新時(shí)間:2023年10月10日 11:33:24   作者:寅恪光潛  
MVCC是一種用于數(shù)據(jù)庫管理系統(tǒng)的并發(fā)控制技術(shù),允許多個(gè)事務(wù)同時(shí)訪問數(shù)據(jù)庫,而不會(huì)導(dǎo)致讀寫沖突,本文就詳細(xì)的介紹了MVCC的具體用法,具有一定的參考價(jià)值,感興趣的可以了解一下

1、概述

MVCC是一種用于數(shù)據(jù)庫管理系統(tǒng)的并發(fā)控制技術(shù),允許多個(gè)事務(wù)同時(shí)訪問數(shù)據(jù)庫,而不會(huì)導(dǎo)致讀寫沖突。也就是說在讀寫的時(shí)候,線程不用去爭(zhēng)搶讀寫鎖。因?yàn)榧渔i的過程比較耗性能。
當(dāng)然很多時(shí)候還是必須的,不能避免,比如說,去ATM機(jī)取錢的時(shí)候,同時(shí)又在手機(jī)APP上進(jìn)行提現(xiàn),這種操作就需要加鎖,不能讓其同時(shí)提現(xiàn),一次只能一個(gè)操作,而且只有在ATM取錢這個(gè)事務(wù)被提交之后才能做其他操作。

處理并發(fā)的場(chǎng)景無外乎三種:

  • 讀、讀:這個(gè)不需要做控制,因?yàn)閿?shù)據(jù)沒有變化
  • 讀、寫:存在線程安全問題,可能出現(xiàn)臟讀、幻讀,不可重復(fù)讀
  • 寫、寫:存在線程安全問題,可能出現(xiàn)更新丟失的情況

這里介紹的MVCC是在存儲(chǔ)引擎為InnoDB實(shí)現(xiàn)的,目的也是為了提高數(shù)據(jù)庫的并發(fā)性能,不使用加鎖的方式去處理讀、寫并發(fā)。

2、MVCC特點(diǎn)

這里的讀操作,有兩種方式:
快照讀:SELECT語句,在讀寫的時(shí)候不用加鎖,所以效率很高,但也存在讀取的時(shí)候有更新操作,可能會(huì)讀到歷史數(shù)據(jù)。

當(dāng)前讀:讀取的是最新數(shù)據(jù),是一種悲觀鎖的操作。它會(huì)對(duì)當(dāng)前讀取的數(shù)據(jù)進(jìn)行加鎖,避免其他事物對(duì)其進(jìn)行寫操作。主要包括以下幾種操作:

  • select lock in share mode(共享鎖)
  • select for update(排他鎖)
  • update(排他鎖)
  • insert(排他鎖)
  • delete(排他鎖)

3、準(zhǔn)備數(shù)據(jù)

在介紹之前,沒有安裝MYSQL的,可以先進(jìn)行安裝,下載地址:https://dev.mysql.com/downloads/
安裝好了之后,我們就新建庫與表,插入一些數(shù)據(jù)來做個(gè)測(cè)試

CREATE DATABASE mydb;
USE mydb;
CREATE TABLE `tb1` (
  `id` int(12) NOT NULL AUTO_INCREMENT,
  `name` varchar(32) DEFAULT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=UTF8;
INSERT INTO tb1(name) VALUES ('XIAO1'),('XIAO2'),('XIAO3');
CREATE TABLE `tb2` (
  `id` int(12) NOT NULL AUTO_INCREMENT,
  `name` varchar(32) DEFAULT NULL,
  PRIMARY KEY (`id`)
) ENGINE=myisam;
INSERT INTO tb2(name) VALUES ('LAO1'),('LAO2'),('LAO3');

這里我特地創(chuàng)建了兩張表,分別是表tb1對(duì)應(yīng)的是InnoDB引擎,表tb2對(duì)應(yīng)的是myisam引擎,創(chuàng)建之后,也可以看到兩者的數(shù)據(jù)格式也是不一樣的,我們先查詢下,新建的數(shù)據(jù)庫mydb以及保存的數(shù)據(jù)在什么地方:

mysql> show variables like '%datadir%';
+---------------+---------------------------------------------+
| Variable_name | Value                                       |
+---------------+---------------------------------------------+
| datadir       | C:\ProgramData\MySQL\MySQL Server 8.0\Data\ |
+---------------+---------------------------------------------+
1 row in set, 1 warning (0.00 sec)

然后可以看到,InnoDB引擎對(duì)應(yīng)的是:tb1.ibd ,myisam對(duì)應(yīng)的是:tb2.MYD(數(shù)據(jù))、tb2.MYI(索引)、tb2_402.sdi(表結(jié)構(gòu))[這個(gè)在以前的版本沒有出現(xiàn)]
MySQL5.5之后都是默認(rèn)為InnoDB引擎

4、MVCC原理

MYSQL存儲(chǔ)的數(shù)據(jù)中,除了我們顯式定義的字段,還隱含著兩個(gè)字段。
trx_id:事務(wù)id,每進(jìn)行一次事務(wù)操作,就會(huì)自增1。
roll_pointer:回滾指針,用于找到上一個(gè)版本的數(shù)據(jù),結(jié)合undolog進(jìn)行回滾。

我們用SELECT讀數(shù)據(jù)時(shí),這一時(shí)刻的數(shù)據(jù)會(huì)有很多個(gè)版本【比如上圖四個(gè)版本】,但我們并不知道讀取哪個(gè)版本,依賴ReadView來對(duì)我們進(jìn)行版本的選擇,通過ReadView我們就能夠知道讀取哪個(gè)版本。我們來看下這個(gè)ReadView的定義:

class ReadView {
/* ... */
private:
trx_id_t m_low_limit_id; /* 大于等于這個(gè) ID 的事務(wù)均不可見 */
trx_id_t m_up_limit_id; /* 小于這個(gè) ID 的事務(wù)均可見 */
trx_id_t m_creator_trx_id; /* 創(chuàng)建該 Read View 的事務(wù)ID */
trx_id_t m_low_limit_no; /* 事務(wù) Number, 小于該 Number 的 Undo Logs 均可以被 Purge */
ids_t m_ids; /* 創(chuàng)建 Read View 時(shí)的活躍事務(wù)列表 */
m_closed; /* 標(biāo)記 Read View 是否 close */
}

字段的解釋:

  • m_low_limit_id:目前出現(xiàn)的最大事務(wù)ID+1(下一個(gè)將被分配的事務(wù)ID)。大于等于這個(gè)ID的數(shù)據(jù)版本均不可見,也就訪問不到。
  • m_up_limit_id:活躍事務(wù)列表m_ids中最小的事務(wù)ID,如果為空,則m_up_limit_id為m_low_limit_id。小于這個(gè)ID的數(shù)據(jù)版本均可見。
  • m_ids:ReadView創(chuàng)建時(shí)其他未提交的活躍事務(wù)ID列表。創(chuàng)建ReadView時(shí),將當(dāng)前未提交事務(wù)ID記錄下來,后續(xù)即使它們修改了記錄行的值,對(duì)于當(dāng)前事務(wù)也是不可見的。m_ids 不包括當(dāng)前事務(wù)自己和已提交的事務(wù)(正在內(nèi)存中)
  • m_creator_trx_id:創(chuàng)建該ReadView的事務(wù)ID

5、實(shí)踐操作

開四個(gè)終端,新建三個(gè)事務(wù),兩個(gè)寫操作,一個(gè)讀操作,還有一個(gè)就是單純的查詢。大家可以根據(jù)不同的進(jìn)入時(shí)間來了解這個(gè)過程,會(huì)產(chǎn)生未提交和已提交事務(wù)狀態(tài),對(duì)應(yīng)的查詢信息都是不一樣的,具體代碼分別如下:

BEGIN;
UPDATE tb1 SET  name='XXX' WHERE id=1;
UPDATE tb1 SET  name='YYY' WHERE id=1;
COMMIT;
BEGIN;
UPDATE tb1 SET  name='ZZZ' WHERE id=1;
UPDATE tb1 SET  name='QQQ' WHERE id=1;
DELETE FROM tb1 WHERE id=5;
COMMIT;
BEGIN;
SELECT * FROM tb1 WHERE id=1;
COMMIT;
SELECT * FROM tb1;

6、小結(jié)

對(duì)于這種讀寫并發(fā),以及對(duì)性能的要求,大家需要看實(shí)際業(yè)務(wù)情況來做決定,其中這里主要是介紹InnoDB引擎,這個(gè)要高效很多,在以前的舊版本可能大家使用MyISAM這個(gè)更多,因?yàn)樾阅芎芎茫贿^不支持事務(wù)操作,所以很多場(chǎng)景也就不適應(yīng),MYSQL5.5版本之后就是默認(rèn)InnoDB引擎了。
最后大家也可以嘗試在表tb2中去嘗試下,看下是什么結(jié)果,因?yàn)檫@張表使用的是MyISAM引擎,就起不到作用了。
另外需要注意的是,事務(wù)的提交是默認(rèn)自動(dòng)的,有些時(shí)候需要關(guān)閉,將默認(rèn)的1修改為0:

SET AUTOCOMMIT=0;
SELECT @@AUTOCOMMIT;

比如說對(duì)于臟讀的情況,我們需要當(dāng)前讀,也就是需要排它鎖:

SET AUTOCOMMIT=0;
BEIGIN;
DELETE FROM tb1 WHERE id=2;

這種情況如果我們的事務(wù)在這個(gè)時(shí)候進(jìn)去,對(duì)這個(gè)id=2進(jìn)行讀寫操作,就會(huì)出現(xiàn)臟讀的情況,這個(gè)時(shí)候就需要使用SELECT FOR UPDATE,等待事務(wù)處理完畢之后再做相應(yīng)的操作。

對(duì)于臟讀這種情況,很常見,比如說事務(wù)在做刪除操作,這個(gè)時(shí)候記錄已被刪除但是還沒有提交事務(wù),如果進(jìn)行查詢操作就會(huì)出現(xiàn)臟讀,如下:

SET AUTOCOMMIT=0;
BEGIN;
SELECT * FROM tb1 WHERE id=2 FOR UPDATE;
UPDATE tb1 SET  name='QQQ' WHERE id=1;
COMMIT;

到此這篇關(guān)于一文詳解MYSQL的多版本并發(fā)控制MVCC(Multi-Version Concurrency Control)的文章就介紹到這了,更多相關(guān)MYSQL多版本并發(fā)控制MVCC內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 詳解mysql的limit經(jīng)典用法及優(yōu)化實(shí)例

    詳解mysql的limit經(jīng)典用法及優(yōu)化實(shí)例

    這篇文章詳細(xì)介紹了mysql的limit經(jīng)典用法及優(yōu)化實(shí)例,有需要的朋友可以參考一下
    2013-09-09
  • MySql常用操作SQL語句匯總

    MySql常用操作SQL語句匯總

    這篇文章主要介紹了MySql常用操作SQL語句匯總,本文講解了增加、刪除、查詢、修改等常用SQL語句實(shí)例,需要的朋友可以參考下
    2015-03-03
  • MySQL之模糊查詢特殊字符處理方式

    MySQL之模糊查詢特殊字符處理方式

    這篇文章主要介紹了MySQL之模糊查詢特殊字符處理方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-08-08
  • 使用JDBC從數(shù)據(jù)庫中查詢數(shù)據(jù)的方法

    使用JDBC從數(shù)據(jù)庫中查詢數(shù)據(jù)的方法

    下面小編就為大家?guī)硪黄褂肑DBC從數(shù)據(jù)庫中查詢數(shù)據(jù)的方法。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2016-08-08
  • 揭秘SQL優(yōu)化技巧 改善數(shù)據(jù)庫性能

    揭秘SQL優(yōu)化技巧 改善數(shù)據(jù)庫性能

    這篇文章是以 MySQL 為背景,很多內(nèi)容同時(shí)適用于其他關(guān)系型數(shù)據(jù)庫,需要有一些索引知識(shí)為基礎(chǔ),重點(diǎn)講述如何優(yōu)化SQL,來提高數(shù)據(jù)庫的性能
    2012-01-01
  • MySQL 存儲(chǔ)過程的基本用法介紹

    MySQL 存儲(chǔ)過程的基本用法介紹

    我們大家都知道MySQL 存儲(chǔ)過程是從 MySQL 5.0 開始逐漸增加新的功能。存儲(chǔ)過程在實(shí)際應(yīng)用中也是優(yōu)點(diǎn)大于缺點(diǎn)。不過最主要的還是執(zhí)行效率和SQL 代碼封裝。特別是 SQL 代碼封裝功能,如果沒有存儲(chǔ)過程。
    2010-12-12
  • MySQL中的聯(lián)合索引學(xué)習(xí)教程

    MySQL中的聯(lián)合索引學(xué)習(xí)教程

    這篇文章主要介紹了MySQL中的聯(lián)合索引學(xué)習(xí)教程,其中談到了聯(lián)合索引對(duì)排序的優(yōu)化等知識(shí)點(diǎn),需要的朋友可以參考下
    2015-11-11
  • mysql 教程 存儲(chǔ)過程

    mysql 教程 存儲(chǔ)過程

    最近用mysql + asp.net來寫網(wǎng)站,既然mysql已經(jīng)支持存儲(chǔ)過程了,那么像分頁這么常用的東西,當(dāng)然要用存儲(chǔ)過程啦
    2009-06-06
  • mysqldump你可能不知道的參數(shù)

    mysqldump你可能不知道的參數(shù)

    這篇文章主要介紹了mysqldump你可能不知道的參數(shù),幫助大家更好的理解和使用MySQL,感興趣的朋友可以了解下
    2020-11-11
  • MySQL配置文件my.cnf中文詳解附mysql性能優(yōu)化方法分享

    MySQL配置文件my.cnf中文詳解附mysql性能優(yōu)化方法分享

    Mysql參數(shù)優(yōu)化對(duì)于新手來講,是比較難懂的東西,其實(shí)這個(gè)參數(shù)優(yōu)化,是個(gè)很復(fù)雜的東西,對(duì)于不同的網(wǎng)站,及其在線量,訪問量,帖子數(shù)量,網(wǎng)絡(luò)情況,以及機(jī)器硬件配置都有關(guān)系,優(yōu)化不可能一次性完成,需要不斷的觀察以及調(diào)試,才有可能得到最佳效果。
    2011-09-09

最新評(píng)論