搜尋
首頁資料庫mysql教程MySQL 一致性读 深入研究_MySQL

 

一致性读,又称为快照读。使用的是MVCC机制读取undo中的已经提交的数据。所以它的读取是非阻塞的。

 

相关文档:http://dev.mysql.com/doc/refman/5.6/en/innodb-consistent-read.html

 

A consistent read means that InnoDB uses multi-versioning to present to a query a snapshot of the database at a point in time. The query sees the changes made by transactions that committed before that point of time, and no changes made by later or uncommitted transactions. The exception to this rule is that the query sees the changes made by earlier statements within the same transaction.

 

一致性读肯定是读取在某个时间点已经提交了的数据,有个特例:本事务中修改的数据,即使未提交的数据也可以在本事务的后面部分读取到。

 

1. RC 隔离 和 RR 隔离中一致性读的区别

 

根据隔离级别的不同,一致性读也是不一样的。不同点在于判断是否提交的“某个时间点”:

 

1)对于RR隔离:

 

If the transaction isolation level is REPEATABLE READ (the default level), all consistent reads within the same transaction read the snapshot established by the first such read in that transaction.

 

文档中说的是:the first such read in that transaction。实际上实验的结果表明,并不是the first such read,而是事务中任何执行的第一条语句为snapshot的起始点,即使该条语句执行失败,也是以它的执行时间为snapshot的起始点。因为事务的起始点其实是以执行的第一条语句为起始点的,而不是以begin作为事务的起始点的。在该起始点之前提交的数据,就可以读取到。(原因应该是RR隔离级别是要支持可重复读的)

 

实验1:

 

sesseion A session B

mysql> set tx_isolation='repeatable-read';

Query OK, 0 rows affected (0.00 sec)

 

mysql> set tx_isolation='repeatable-read';

Query OK, 0 rows affected (0.00 sec)

 

mysql> begin;

Query OK, 0 rows affected (0.01 sec)

 
 

mysql> select * from t1;

Empty set (0.00 sec)

 

mysql> insert into t1(c1,c2) values(1,1);

Query OK, 1 row affected (0.01 sec)

mysql> select * from t1;

+----+------+

| c1 | c2   |

+----+------+

|  1 |    1 |

+----+------+

1 row in set (0.00 sec)

 

 

上面的实验说明:RR隔离级别下的一致性读,不是以begin开始的时间点作为snapshot建立时间点,而是以第一条语句的时间点作为snapshot建立的时间点。

 

实验2:

 

session A session B
mysql> set tx_isolation='repeatable-read'; mysql> set tx_isolation='repeatable-read';
 

mysql> select * from t1;

Empty set (0.00 sec)

mysql> begin;

mysql> set i=1;

ERROR 1193 (HY000): Unknown system variable 'i'

 
 

mysql> insert into t1(c1,c2) values(1,1);

Query OK, 1 row affected (0.01 sec)

mysql> select * from t1;

+----+------+

| c1 | c2   |

+----+------+

|  1 |    1 |

+----+------+

1 row in set (0.00 sec)

 

 

 

该使用说明:RR隔离级别下的一致性读,是以第一语句的执行点作为snapshot建立的时间点的,即使该语句执行失败了,也是如此。

 

实验3:

 

session A session B
mysql> set tx_isolation='repeatable-read';

mysql> set tx_isolation='repeatable-read';

mysql> select * from t1;

Empty set (0.00 sec)

mysql> begin;  

mysql> select * from t1;

Empty set (0.00 sec)

mysql> select * from t1;

Empty set (0.00 sec)

  mysql> insert into t1(c1,c2) values(1,1);

mysql> select * from t1;

Empty set (0.01 sec)

 

该实验中:session A 的第一条语句,发生在session B的 insert语句提交之前,所以session A中的第二条select还是不能读取到数据。因为RR中的一致性读是以事务中第一个语句执行的时间点作为snapshot建立的时间点的。而此时,session B的insert语句还没有执行,所以读取不到数据。

 

实验4:

 

session A session B
mysql> set tx_isolation='repeatable-read';

mysql> set tx_isolation='repeatable-read';

mysql> select * from t1;

Empty set (0.00 sec)

mysql> select * from t1;

Empty set (0.00 sec)

 
 

mysql> insert into t1(c1,c2) values(1,1),(2,2);

mysql> select * from t1;

+----+------+

| c1 | c2   |

+----+------+

|  1 |    1 |

|  2 |    2 |

+----+------+

2 rows in set (0.01 sec)

mysql> select * from t1;

Empty set (0.00 sec)

 

mysql> update t1 set c2=100 where c1=1;

Query OK, 1 row affected (0.00 sec)

Rows matched: 1  Changed: 1  Warnings: 0

 

mysql> select * from t1;

+----+------+

| c1 | c2   |

+----+------+

|  1 |  100 |

+----+------+

1 row in set (0.00 sec)

 

 

该实验说明:本事务中进行修改的数据,即使没有提交,在本事务中的后面也可以读取到。update 语句因为进行的是“当前读”,所以它可以修改成功。

 

2)对于RC隔离就简单多了:

 

With READ COMMITTED isolation level, each consistent read within a transaction sets and reads its own fresh snapshot.

 

事务中每一次读取都是以当前的时间点作为判断是否提交的实际点,也即是 reads its own fresh snapshot.

 

RC是语句级多版本(事务的多条只读语句,创建不同的ReadView,代价更高),RR是事务级多版本(一个ReadView);

 

2. Oracle中的一致性读

 

Oracle读一致性是指一个查询所获得的数据来自同一时间点。

 

Oracle读一致性分为语句级读一致性和事务级读一致性。

 

语句级读一致性:Oracle强制实现语句级读一致性。一个查询语句只读取语句开始之前提交的数据。

 

事务级读一致性:隔离级别为SERIALIZABLE和read only的事务才支持事务级读一致性。事务中的所有查询语句只读取 事务开始之前提交的数据。

 

Oracle只实现了RC和serializable,没有实现Read uncommitted 和 RR。其实Oracle的serializable级别才实现了RR可重复读。

 

3. 当前读(current read) 和 一致性读

 

一致性读是指普通的select语句,不带 for update, in share mode 等等子句。使用的是undo中的提交的数据,不需要使用锁(MDL除外)。而当前读,是指update, delete, select for update, select in share mode等等语句进行的读,它们读取的是数据库中的最新的数据,并且会锁住读取的行和gap(RR隔离时)。如果不能获得锁,则会一直等待,直到获得或者超时。

 

4. 一致性读与 mysqldump --single-transaction

 

我们知道 mysqldump --single-transaction的原理是:设置事务为RR模式,然后利用事务的特性,来获得一致性的数据,但是:

 

--single-transaction

                      Creates a consistent snapshot by dumping all tables in a

                      single transaction. Works ONLY for tables stored in

                      storage engines which support multiversioning (currently

                      only InnoDB does); the dump is NOT guaranteed to be

                      consistent for other storage engines. While a

                      --single-transaction dump is in process, to ensure a

                      valid dump file (correct table contents and binary log

                      position), no other connection should use the following

                      statements: ALTER TABLE, DROP TABLE, RENAME TABLE,

                      TRUNCATE TABLE, as consistent snapshot is not isolated

                      from them. Option automatically turns off --lock-tables.

 

在mysqldump运行期间,不能执行 alter table, drop table, rename table, truncate table 等等的DDL语句,因为一致性读和这些语句时无法隔离的。

 

那么在mysqldump --single-transaction 执行期间,执行了上面那些DDL,会发生什么呢?

 

mysqldump --single-transaction 的执行过程是:设置RR,然后开始事务,对应了一个LSN,然后对所有选中的表,一个一个的执行下面的过程:

 

save point sp; --> select * from t1 --> rollback to sp;

 

save point sp; --> select * from t2 --> rollback to sp;

 

... ...

 

1> 那么如果对t2表的DDL发生在 save point sp 之前,那么当mysqldump处理到 t2 表时,mysqldump 会立马报错:表结构已经改变......

 

2> 如果对t2表的DDL发生在 save point sp 之后,rollback to sp 之前,那么要么DDL被阻塞,要么mysqldump被阻塞,具体谁被阻塞,看谁先执行了。

 

     被阻塞额原因是:DDL需要t2表的 MDL 的互斥锁,而select * from t1 需要MDL的共享锁,所以阻塞发生。

 

3> 如果对t2表的DDL发生在 rollback to sp 之后,那么因为对 t2 表的dump已经完成,不会发生错误或者阻塞。

 

那么为什么: 对t2表的DDL发生在 save point sp 之前,那么当mysqldump开始处理 t2 表时,mysqldump 立马报错呢?

 

其原因就是 一致性读的胳膊拗不过DDL的大腿:

 

Consistent read does not work over certain DDL statements:(一致性读的胳膊拗不过DDL的大腿)

 

Consistent read does not work over DROP TABLE, because MySQL cannot use a table that has been dropped and InnoDB destroys the table.

 

Consistent read does not work over ALTER TABLE, because that statement makes a temporary copy of the original table and deletes the original table when the temporary copy is built. When you reissue a consistent read within a transaction, rows in the new table are not visible because those rows did not exist when the transaction's snapshot was taken. In this case, the transaction returns an error as of MySQL 5.6.6: ER_TABLE_DEF_CHANGED, “Table definition has changed, please retry transaction”.

 

原因:ALTER TABLE, DROP TABLE, RENAME TABLE, TRUNCATE TABLE 这些DDL语句的执行,会导致无法使用undo构造出正确的一致性读,一致性读和它们是无法隔离的。

陳述
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn
Redis实现分布式事务的一致性与可靠性保障Redis实现分布式事务的一致性与可靠性保障Jun 20, 2023 am 09:00 AM

Redis是一个开源的、高性能的NoSQL数据库,由于其快速读写速度、数据结构比较丰富,被广泛应用于缓存、队列和分布式锁等方面。但是,在分布式事务领域的应用还有待深入研究。本文将从Redis的特点出发,探讨如何利用Redis实现分布式事务的一致性与可靠性保障。一、Redis的数据结构特点Redis支持的数据结构非常丰富,包括字符串、列表、哈希表、集合等等。这

如何在Java中实现分布式缓存的一致性和容错机制如何在Java中实现分布式缓存的一致性和容错机制Oct 09, 2023 pm 06:27 PM

如何在Java中实现分布式缓存的一致性和容错机制分布式缓存是互联网高并发系统中常用的技术,它可以提高系统的性能和扩展性。然而,分布式缓存面临着一致性和容错的挑战。在本文中,我们将讨论如何在Java中实现分布式缓存的一致性和容错机制,并提供具体的代码示例。一、一致性机制在分布式环境下,缓存的一致性是非常重要的。分布式缓存的一致性可以通过以下两种机制实现:缓存更

Go语言中如何处理并发数据库数据一致性问题?Go语言中如何处理并发数据库数据一致性问题?Oct 10, 2023 pm 03:37 PM

Go语言中如何处理并发数据库数据一致性问题?当多个并发请求同时访问数据库时,会引发数据一致性问题。在Go语言中,我们可以使用事务和锁来处理这个问题。下面我将详细介绍如何在Go语言中处理并发数据库数据一致性问题,并给出具体的代码示例。首先,我们需要使用数据库的事务机制。数据库事务提供了一种机制,用于将一系列的数据库操作看作是一个整体,要么全部成功,要么全部失败

Redis在分布式事务的可靠性与一致性对比Redis在分布式事务的可靠性与一致性对比Jun 20, 2023 am 09:38 AM

随着互联网应用的快速发展,分布式架构已经成为了企业级应用的重要选择。而作为其中一种常见的缓存技术,Redis也扮演着重要的角色。分布式事务的可靠性与一致性是架构设计中不可避免的话题之一,本文将以Redis为例,探讨其在分布式事务中的可靠性与一致性对比。一、Redis的常见问题Redis通过将数据缓存在内存中,提供快速、高效的访问。但同时也因此面临着诸如数据丢

如何解决Java功能开发中的分布式一致性问题如何解决Java功能开发中的分布式一致性问题Aug 08, 2023 pm 08:53 PM

如何解决Java功能开发中的分布式一致性问题在当今互联网应用的开发中,分布式架构已经成为一种常见的技术选型。与传统的单体应用相比,分布式系统具有高可用性、高性能和可扩展性等诸多优势。然而,分布式应用的开发也面临着一系列挑战,其中之一就是分布式的一致性问题。在分布式系统中,不同的服务节点并不总是能够即时地达成一致的状态。由于网络延迟、节点故障和并发更新等原因,

利用MySQL开发实现分布式事务与一致性的项目经验探讨利用MySQL开发实现分布式事务与一致性的项目经验探讨Nov 03, 2023 am 11:18 AM

利用MySQL开发实现分布式事务与一致性的项目经验探讨引言:随着互联网行业的快速发展,分布式架构成为了很多大型系统和应用的首选。在分布式环境下,事务和一致性成为了开发者面临的重要挑战之一。本文将结合一个实际项目的经验,探讨如何利用MySQL开发实现分布式事务与一致性。一、背景和问题:我们的团队负责开发一套分布式电商平台,该平台需要支持用户的下单、支付和库存扣

Redis作为缓存数据库的一致性策略Redis作为缓存数据库的一致性策略Jun 21, 2023 am 11:03 AM

Redis作为缓存数据库已经变得越来越流行,为了保证应用程序的高可用性和高性能,应该采用一致性策略来确保数据的一致性和可靠性。一致性策略是指应用程序、持久化存储和缓存数据库之间的数据一致性。在分布式系统中,由于计算机之间的消息传递和同步机制,可能会导致数据的不一致性。因此,我们需要采用一致性策略来避免这种情况发生。Redis作为缓存数据库的一致性策略主要包括

Redis实现分布式缓存一致性的方法与应用实例Redis实现分布式缓存一致性的方法与应用实例May 11, 2023 pm 02:40 PM

Redis是一种高性能的内存数据库,被广泛应用于大规模数据的存储和处理。除了作为独立的数据库使用之外,Redis还可以作为缓存层,用来提高系统的访问速度。在分布式应用场景中,Redis作为分布式缓存的角色越来越重要。然而,在分布式环境下,如何保证Redis分布式缓存的一致性是开发人员需要面对的难题。本文将介绍Redis实现分布式缓存一致性的方法与应用实例。一

See all articles

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

AI Hentai Generator

AI Hentai Generator

免費產生 AI 無盡。

熱門文章

R.E.P.O.能量晶體解釋及其做什麼(黃色晶體)
2 週前By尊渡假赌尊渡假赌尊渡假赌
倉庫:如何復興隊友
4 週前By尊渡假赌尊渡假赌尊渡假赌
Hello Kitty Island冒險:如何獲得巨型種子
3 週前By尊渡假赌尊渡假赌尊渡假赌

熱工具

WebStorm Mac版

WebStorm Mac版

好用的JavaScript開發工具

SublimeText3 Linux新版

SublimeText3 Linux新版

SublimeText3 Linux最新版

ZendStudio 13.5.1 Mac

ZendStudio 13.5.1 Mac

強大的PHP整合開發環境

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

SublimeText3 英文版

SublimeText3 英文版

推薦:為Win版本,支援程式碼提示!