ホームページ データベース mysql チュートリアル MySQL 一致性读 深入研究_MySQL

MySQL 一致性读 深入研究_MySQL

May 30, 2016 pm 05:10 PM
一貫性

 

一致性读,又称为快照读。使用的是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 までご連絡ください。

ホットAIツール

Undresser.AI Undress

Undresser.AI Undress

リアルなヌード写真を作成する AI 搭載アプリ

AI Clothes Remover

AI Clothes Remover

写真から衣服を削除するオンライン AI ツール。

Undress AI Tool

Undress AI Tool

脱衣画像を無料で

Clothoff.io

Clothoff.io

AI衣類リムーバー

AI Hentai Generator

AI Hentai Generator

AIヘンタイを無料で生成します。

ホットツール

メモ帳++7.3.1

メモ帳++7.3.1

使いやすく無料のコードエディター

SublimeText3 中国語版

SublimeText3 中国語版

中国語版、とても使いやすい

ゼンドスタジオ 13.0.1

ゼンドスタジオ 13.0.1

強力な PHP 統合開発環境

ドリームウィーバー CS6

ドリームウィーバー CS6

ビジュアル Web 開発ツール

SublimeText3 Mac版

SublimeText3 Mac版

神レベルのコード編集ソフト(SublimeText3)

Redisは分散トランザクションの一貫性と信頼性保証を実現します Redisは分散トランザクションの一貫性と信頼性保証を実現します Jun 20, 2023 am 09:00 AM

Redis は、オープンソースの高性能 NoSQL データベースであり、その高速な読み取りおよび書き込み速度と豊富なデータ構造により、キャッシュ、キュー、分散ロックなどで広く使用されています。ただし、分散トランザクションの分野での応用については、さらに研究する必要があります。この記事では、Redis の特徴から始めて、分散トランザクションの一貫性と信頼性を確保するために Redis を使用する方法について説明します。 1. Redis のデータ構造の特徴 Redis は、文字列、リスト、ハッシュ テーブル、セットなど、幅広いデータ構造をサポートしています。これ

Java で分散キャッシュの一貫性とフォールト トレランスのメカニズムを実装する方法 Java で分散キャッシュの一貫性とフォールト トレランスのメカニズムを実装する方法 Oct 09, 2023 pm 06:27 PM

Java で分散キャッシュの一貫性とフォールトトレランスのメカニズムを実装する方法分散キャッシュは、同時実行性の高いインターネット システムで一般的に使用されるテクノロジであり、システムのパフォーマンスとスケーラビリティを向上させることができます。ただし、分散キャッシュは一貫性と耐障害性の課題に直面しています。この記事では、Java で分散キャッシュの一貫性とフォールト トレランスを実装する方法について説明し、具体的なコード例を示します。 1. 一貫性のメカニズム 分散環境では、キャッシュの一貫性が非常に重要です。分散キャッシュの一貫性は、次の 2 つのメカニズムを通じて実現できます。 キャッシュの更新

Go 言語で同時データベースのデータ整合性の問題に対処するにはどうすればよいですか? Go 言語で同時データベースのデータ整合性の問題に対処するにはどうすればよいですか? Oct 10, 2023 pm 03:37 PM

Go 言語で同時データベースのデータ整合性の問題に対処するにはどうすればよいですか?複数の同時リクエストが同時にデータベースにアクセスすると、データの一貫性の問題が発生する可能性があります。 Go 言語では、トランザクションとロックを使用してこの問題に対処できます。以下では、Go 言語で同時データベースのデータ整合性の問題を処理する方法と、具体的なコード例を詳しく紹介します。まず、データベースのトランザクション メカニズムを使用する必要があります。データベース トランザクションは、一連のデータベース操作 (すべて成功するかすべて失敗するか) を全体として扱うためのメカニズムを提供します。

分散トランザクションにおける Redis の信頼性と一貫性の比較 分散トランザクションにおける Redis の信頼性と一貫性の比較 Jun 20, 2023 am 09:38 AM

インターネット アプリケーションの急速な発展に伴い、分散アーキテクチャはエンタープライズ レベルのアプリケーションにとって重要な選択肢となっています。一般的なキャッシュ テクノロジの 1 つとして、Redis も重要な役割を果たします。分散トランザクションの信頼性と一貫性は、アーキテクチャ設計において避けられないトピックの 1 つです。この記事では、Redis を例として、分散トランザクションの信頼性と一貫性の比較について説明します。 1. Redis に関する一般的な問題 Redis は、データをメモリにキャッシュすることにより、高速かつ効率的なアクセスを提供します。しかし同時に、データ損失などの問題にも直面しています。

MySql と分散トランザクション: 分散データの一貫性を扱う方法 MySql と分散トランザクション: 分散データの一貫性を扱う方法 Jun 15, 2023 pm 09:20 PM

インターネット技術の発展に伴い、分散アプリケーション システムはプログラマーが日常業務で直面しなければならない課題となっています。分散データを扱う場合、一貫性を確保することが最大の懸念事項の 1 つです。現時点では、分散アプリケーションに必要なほとんどの機能を提供できる MySql が好まれるソリューションです。この記事では、MySql を使用して分散環境におけるデータの一貫性の問題を解決する方法を紹介します。分散トランザクションとは何ですか? 分散トランザクションとは、トランザクションの操作に複数の独立した計算が含まれることを意味します。

Java 関数開発における分散一貫性の問題を解決する方法 Java 関数開発における分散一貫性の問題を解決する方法 Aug 08, 2023 pm 08:53 PM

Java 関数開発における分散一貫性の問題を解決する方法 今日のインターネット アプリケーションの開発では、分散アーキテクチャが一般的なテクノロジの選択となっています。従来のモノリシック アプリケーションと比較して、分散システムには高可用性、高性能、拡張性などの多くの利点があります。ただし、分散アプリケーションの開発は一連の課題にも直面しており、そのうちの 1 つは分散の一貫性です。分散システムでは、さまざまなサービス ノードが常に瞬時に一貫した状態に達するとは限りません。ネットワークの遅延、ノードの障害、同時更新などが原因です。

MySQL を使用して分散トランザクションと一貫性を開発したプロジェクトの経験についてのディスカッション MySQL を使用して分散トランザクションと一貫性を開発したプロジェクトの経験についてのディスカッション Nov 03, 2023 am 11:18 AM

MySQL を使用して分散トランザクションと一貫性を開発したプロジェクトの経験についてのディスカッション はじめに: インターネット業界の急速な発展に伴い、分散アーキテクチャは多くの大規模システムやアプリケーションにとって最初の選択肢になりました。分散環境では、トランザクションと一貫性が開発者が直面する重要な課題の 1 つになっています。この記事では、実際のプロジェクトの経験を組み合わせて、MySQL 開発を使用して分散トランザクションと一貫性を実現する方法を検討します。 1. 背景と問題点: 私たちのチームは、ユーザーの注文、支払い、在庫控除をサポートする必要がある分散型電子商取引プラットフォームの開発を担当しています。

キャッシュ データベースの一貫性戦略としての Redis キャッシュ データベースの一貫性戦略としての Redis Jun 21, 2023 am 11:03 AM

Redis はキャッシュ データベースとしてますます人気が高まっていますが、アプリケーションの高可用性と高パフォーマンスを確保するには、データの一貫性と信頼性を確保する一貫性戦略を採用する必要があります。整合性ポリシーは、アプリケーション、永続ストレージ、キャッシュ データベース間のデータの整合性を指します。分散システムでは、コンピュータ間のメッセージ受け渡しおよび同期メカニズムが原因で、データの不整合が発生する可能性があります。したがって、このような事態を避けるためには、一貫性戦略を採用する必要があります。キャッシュ データベースとしての Redis の一貫性戦略には主に次のものがあります。

See all articles