読み取りの分離、整合性、最新性
隔離保証
読み取りコミットなし
読み取り保証(read concern)に応じて、クライアントは、書込みが 耐久性を持つ前に、書込みの結果を確認できます。
書込み(write)の書込み保証(write concern)に関係なく、
"local"
または"available"
の読み取り保証(read concern)を使用する他のクライアントは、書込み(write)操作が発行クライアントに確認される前に、書込み(write)操作の結果を確認できます。"local"
または"available"
読み取り保証(read concern)を使用するクライアントにより、後でレプリカセットのフェイルオーバー中にロールバックされる可能性があるデータを読み取れます。
マルチドキュメントトランザクションの操作では、トランザクションがコミットされると、トランザクション内のすべてのデータ変更が保存され、トランザクションの外部に表示されます。つまり、トランザクションが他の変更をロールバックしながら、一部の変更をコミットすることはありません。
トランザクションがコミットされるまで、トランザクションで行われたデータ変更はトランザクションの外部には表示されません。
ただし、トランザクションが複数のシャードに書き込む場合、すべての外部読み取り操作が、コミットされたトランザクションの結果がシャード全体で表示されるまで待機する必要はありません。たとえば、トランザクションがコミットされ、書込み 1 がシャード A で表示されているものの、書込み 2 がシャード B にまだ表示されていない場合、読み取り保証(read concern) "local"
での外部読み取りは、書き込み 2 を見ることなく書き込み 1 の結果を読み取ることができます。
読み取りコミットなしとは、デフォルトの分離レベルであり、 mongod
スタンドアロン インスタンスだけでなく、レプリカセットとシャーディングされたクラスターにも適用されます。
読み取りコミットなしと単一ドキュメントの不可分性
書き込み操作は単一のドキュメントに対してアトミックです。つまり、書き込みがドキュメント内の複数のフィールドを更新する場合、読み取り操作では一部のフィールドのみが更新されたドキュメントが表示されることはありません。ただし、 クライアントには部分的にアップデートされたドキュメントは表示されない場合がありますが 、読み取りコミットがない場合は、同時に実行された読み取り操作でも、変更が永続化される前に更新されたドキュメントが表示される可能性があります。
スタンドアロンの mongod
インスタンスでは、1 つのドキュメントに対する一連の読み取り操作と書込み操作をシリアル化できます。レプリカセットでは、1つのドキュメントに対する一連の読み取りおよび書込み操作をシリアル化できるのは、ロールバックがない場合だけです。
読み取りコミットなしと複数ドキュメントの書込み
単一の書込み操作(例: db.collection.updateMany()
)は複数のドキュメントを変更します。各ドキュメントの変更は不可分ですが、操作全体は不可分ではありません。
複数のドキュメントへの書込み操作を実行する場合、1 回の書込み操作でも複数の書込み操作でも、他の操作がインターリーブすることがあります。
MongoDB は(単一または複数のコレクション内の)複数のドキュメントへの読み取りと書込みにアトミック性を必要とする状況で、レプリカセットやシャーディングされたクラスターでのトランザクションを含む分散トランザクションをサポートします。
詳細についてはトランザクションを参照してください。
重要
ほとんどの場合、分散トランザクションでは 1 つのドキュメントの書き込み (write) よりもパフォーマンス コストが高くなります。分散トランザクションの可用性は、効果的なスキーマ設計の代わりにはなりません。多くのシナリオにおいて、非正規化されたデータモデル(埋め込みドキュメントと配列)が引き続きデータやユースケースに最適です。つまり、多くのシナリオにおいて、データを適切にモデリングすることで、分散トランザクションの必要性を最小限に抑えることができます。
トランザクションの使用に関するその他の考慮事項(ランタイム制限や oplog サイズ制限など)については、「本番環境での考慮事項」も参照してください。
マルチドキュメントの書込み操作を分離しない場合、MongoDB は次の動作を示します。
非ポイント・イン・タイム読み取り操作。ドキュメントの読み取り操作が時刻t 1に開始されるとします。その後、書込み操作によって、後の時間 t 2 にドキュメントの 1 つに対する更新がコミットされます。読者はドキュメントの更新されたバージョンを参照できますが、データの特定時点のスナップショットは参照できません。
シリアル化できない操作。読み取り操作によってドキュメント d 1 が時刻 t 1 に読み取られ、書込み操作によってその後の時刻 t 3 に d 1 が更新されるとします。これにより、読み取りと書込みの依存関係が導入され、操作をシリアル化する場合、読み取り操作が書込み操作に先行する必要があります。しかし、書込み操作によってドキュメント d 2 が時刻 t 2 に更新され、その後読み取り操作によって時刻 t 4 に d 2 が読み取られるとします。これにより、書込みと読み取りの依存関係が導入され、シリアル化可能なスケジュールで読み取り操作が書込み操作の後に実行されることが必要になります。依存関係のサイクルが存在するため、シリアル化が不可能になります。
読み取り操作の過程で更新された一致するドキュメントが読み取られない場合があります。
カーソルのスナップショット
MongoDB カーソルは、状況によっては、同じドキュメントを複数回返すことがあります。カーソルによってドキュメントが返されると、他の操作がクエリとインターリーブする可能性があります。これらの操作のいずれかによって、クエリで使用されるインデックスのインデックス付きフィールドが変更されると、カーソルは同じドキュメントを複数回返す可能性があります。
ユニークインデックスを使用するクエリでは、重複した値が返される場合があります。ユニークインデックスを使用するカーソルが、同じユニークな値を共有するドキュメントの削除と挿入をインターリーブする場合、カーソルは異なるドキュメントから同じユニークな値を 2 回返すことがあります。
読み取り分離の使用を検討してください。詳しくは、「懸念事項を読む"snapshot"
」を参照してください。
単調書込み
MongoDB は、スタンドアロンの mongod
インスタンスとレプリカ セットに対して、デフォルトで単調な書込み保証を提供します。
単調な書込みとシャーディングされたクラスターについては、 因果整合性を参照してください。
リアルタイム注文
プライマリでの読み取りおよび書込み操作の場合、"linearizable"
読み取り保証(read concern)を使用して読み取り操作を発行し、"majority"
書込み保証(write concern)を使用して書込み操作を発行すると、1 つのスレッドがこれらの操作をリアルタイムで実行したかのように、複数のスレッドが 1 つのドキュメントに対して読み取りおよび書込みを実行できるようになります。つまり、これらの読み取りおよび書込みの対応するスケジュールは線形化可能とみなされます。
因果整合性
ある操作が先行する操作に論理的に依存する場合、それらの操作の間には因果関係があります。たとえば、指定された条件に基づいてすべてのドキュメントを削除する書込み操作と、削除操作を検証する後続の読み取り操作には因果関係があります。
因果的に一貫性のあるセッションでは、MongoDB は因果関係を尊重する順序で因果操作を実行し、クライアントは因果関係と一貫性のある結果を観察します。
クライアント セッションと因果整合性の保証
因果一貫性を提供するために、MongoDB はクライアント セッションで因果一貫性を有効にします。因果的に一貫性のあるセッションとは、"majority"
読み取りコンテントを持つ読み取り操作の関連シーケンスと、"majority"
書込みコンテントを持つ書込み操作の関連シーケンスに、順序によって反映される因果関係があることを示します。アプリケーションでは、クライアント セッションで一度に 1 つのスレッドのみがこれらの操作が実行されるようにする必要があります。
因果関係のある操作の場合:
クライアントがクライアント セッションを開始します。
重要
クライアント セッションでは、次の因果整合性のみが保証されます。
"majority"
を使用した読み取り操作、つまり、返されたデータは、レプリカ セット ノードの大多数によって確認されており、耐久性があります。"majority"
書込み保証(write concern)付き書込み(write)操作。つまり、レプリカセットの投票ノードの過半数に操作が適用済みであることを確認するようリクエストする書込み操作です。
因果関係の一貫性と読み取りと書込み保証(write concern)に関するさまざまな問題について詳しくは、「因果整合性、読み取り保証、書込み保証」を参照してください。
クライアントが、
"majority"
読み取り保証(read concern)を持つ読み取り操作と"majority"
書込み保証(write concern)を持つ書込み操作のシーケンスを発行するため、クライアントは各操作にセッション情報を含めることができます。セッションに関連付けられた
"majority"
読み取り保証(read concern)持つ読み取り操作と"majority"
書込み保証(write concern)を持つ書込み操作ごとに、操作にエラーがあった場合でも、MongoDB は操作時間とクラスター時間を返します。クライアント セッションは操作時間とクラスター時間を記録します。注意
MongoDB では、未確認の(
w: 0
)書込み操作の操作時間とクラスター時間が返されません。未確認の書込みは、因果関係を意味するものではありません。MongoDB は、クライアント セッションでの読み取り操作と確認済みの書き込み操作の操作時間とクラスター時間を返しますが、因果一貫性を保証できるのは、
"majority"
読み取り保証(read concern)を持つ読み取り操作と"majority"
書込み保証(write concern)を持つ書き込み操作のみです。詳細については、「因果整合性、読み取り保証、書込み保証」を参照してください。関連付けられているクライアント セッションは、これら 2 つの時間フィールドを追跡します。
注意
異なるセッション間でも、操作は因果的に一貫している場合があります。MongoDB ドライバーと
mongosh
、クライアント セッションの操作時間とクラスター時間を進めるメソッドを提供します。そのため、クライアントは、1 つのクライアント セッションのクラスター時間と操作時間を、別のクライアント セッションの操作と一致するように進めることができます。
因果整合性の保証
次の表は、"majority"
読み取り保証(read concern)を持つ読み取り操作と "majority"
書込み保証(write concern)を持つ書込み操作に対して因果的に整合性のあるセッションによって提供される因果整合性の保証をリスとしたものです。
保証 | 説明 |
---|---|
書込み操作の結果を読み取る | 読み取り操作は、その前の書込み操作の結果を反映します。 |
単調な読み取り | 読み取り操作では、前の読み取り操作よりも前のデータの状態に対応する結果は返されません。 たとえば、セッション中の場合:
そのため、読み取り2は書込み1の結果を返すことができません。 |
単調書込み | 他の書込みに優先する必要がある書込み操作は、他の書込みの前に実行する必要があります。 たとえば、セッションで書込み1が書込み2に優先する必要がある場合、書込み2の時点でのデータの状態は、書込み1後のデータの状態を反映している必要があります。書込み1と書込み2の間で他の書込みをインターリーブすることはできますが 、 書込み1の前に書込み2を行うことはできません 。 |
書込み操作の前に読み取り操作をする | 読み取り操作の後に実行する必要がある書込み操作は、それらの読み取り操作の後に実行されます。つまり、書込み時のデータの状態には、前の読み取り操作のデータの状態が反映されている必要があります。 |
読み込み設定 (read preference)
これらの保証は、MongoDB 配置のすべてのノードに適用されます。たとえば、因果関係が整合しているセッションで、"majority"
書込み保証(write concern)が懸念される書込みを実行した後に、セカンダリからの読み取りを実行した場合(つまり、読み込み設定(read preferencesecondary
)と読み取り保証(read concern)"majority"
の場合、読み取り操作は書込み操作後のデータベースの状態を反映します。
分離
因果的に整合したセッション内の操作は、セッション外の操作から分離されません。セッションの書込み操作と読み取り操作の間に同時書込み操作が交互に行われる場合、セッションの読み取り操作は、セッションの書込み操作の後に発生した書込み操作を反映した結果を返すことがあります。
MongoDB ドライバー
Tip
アプリケーションでは、クライアント セッションで一度に 1 つのスレッドのみがこれらの操作が実行されるようにする必要があります。
クライアントには、MongoDB 3.6 以降用の最新 MongoDB ドライバーが必要です。
Java 3.6+ Python 3.6+ C 1.9+ Go 1.8+ | C# 2.5+ Node 3.0+ Ruby 2.5+ Rust 2.1+ Swift 1.2+ | Perl 2.0+ PHPC 1.4+ Scala 2.2+ C++ 3.6.6+ |
例
重要
因果的に整合性のあるセッションでは、 "majority"
読み取り保証(read concern)のある読み取りと "majority"
書込み保証(write concern)のある書込みに対してのみ因果整合性が保証されます。
さまざまなアイテムの現在のデータと履歴データを保持するコレクションitems
を検討します。履歴データのみに null 以外の end
日付があります。アイテムの sku
値が変更された場合は、古い sku
値を持つドキュメントを end
日付で更新し、その後、現在の sku
値を持つ新しいドキュメントを挿入する必要があります。クライアントは因果的に一貫したセッションを使用して、挿入の前に更新が行われるようにすることができます。
➤ 右上の [言語の選択] ドロップダウン メニューを使用して、この例の言語を設定します。
/* Use a causally-consistent session to run some operations. */ wc = mongoc_write_concern_new (); mongoc_write_concern_set_wmajority (wc, 1000); mongoc_collection_set_write_concern (coll, wc); rc = mongoc_read_concern_new (); mongoc_read_concern_set_level (rc, MONGOC_READ_CONCERN_LEVEL_MAJORITY); mongoc_collection_set_read_concern (coll, rc); session_opts = mongoc_session_opts_new (); mongoc_session_opts_set_causal_consistency (session_opts, true); session1 = mongoc_client_start_session (client, session_opts, &error); if (!session1) { fprintf (stderr, "couldn't start session: %s\n", error.message); goto cleanup; } /* Run an update_one with our causally-consistent session. */ update_opts = bson_new (); res = mongoc_client_session_append (session1, update_opts, &error); if (!res) { fprintf (stderr, "couldn't add session to opts: %s\n", error.message); goto cleanup; } query = BCON_NEW ("sku", "111"); update = BCON_NEW ("$set", "{", "end", BCON_DATE_TIME (bson_get_monotonic_time ()), "}"); res = mongoc_collection_update_one (coll, query, update, update_opts, NULL, /* reply */ &error); if (!res) { fprintf (stderr, "update failed: %s\n", error.message); goto cleanup; } /* Run an insert with our causally-consistent session */ insert_opts = bson_new (); res = mongoc_client_session_append (session1, insert_opts, &error); if (!res) { fprintf (stderr, "couldn't add session to opts: %s\n", error.message); goto cleanup; } insert = BCON_NEW ("sku", "nuts-111", "name", "Pecans", "start", BCON_DATE_TIME (bson_get_monotonic_time ())); res = mongoc_collection_insert_one (coll, insert, insert_opts, NULL, &error); if (!res) { fprintf (stderr, "insert failed: %s\n", error.message); goto cleanup; }
using (var session1 = client.StartSession(new ClientSessionOptions { CausalConsistency = true })) { var currentDate = DateTime.UtcNow.Date; var items = client.GetDatabase( "test", new MongoDatabaseSettings { ReadConcern = ReadConcern.Majority, WriteConcern = new WriteConcern( WriteConcern.WMode.Majority, TimeSpan.FromMilliseconds(1000)) }) .GetCollection<BsonDocument>("items"); items.UpdateOne(session1, Builders<BsonDocument>.Filter.And( Builders<BsonDocument>.Filter.Eq("sku", "111"), Builders<BsonDocument>.Filter.Eq("end", BsonNull.Value)), Builders<BsonDocument>.Update.Set("end", currentDate)); items.InsertOne(session1, new BsonDocument { {"sku", "nuts-111"}, {"name", "Pecans"}, {"start", currentDate} }); }
// Example 1: Use a causally consistent session to ensure that the update occurs before the insert. ClientSession session1 = client.startSession(ClientSessionOptions.builder().causallyConsistent(true).build()); Date currentDate = new Date(); MongoCollection<Document> items = client.getDatabase("test") .withReadConcern(ReadConcern.MAJORITY) .withWriteConcern(WriteConcern.MAJORITY.withWTimeout(1000, TimeUnit.MILLISECONDS)) .getCollection("test"); items.updateOne(session1, eq("sku", "111"), set("end", currentDate)); Document document = new Document("sku", "nuts-111") .append("name", "Pecans") .append("start", currentDate); items.insertOne(session1, document);
async with await client.start_session(causal_consistency=True) as s1: current_date = datetime.datetime.today() items = client.get_database( "test", read_concern=ReadConcern("majority"), write_concern=WriteConcern("majority", wtimeout=1000), ).items await items.update_one( {"sku": "111", "end": None}, {"$set": {"end": current_date}}, session=s1 ) await items.insert_one( {"sku": "nuts-111", "name": "Pecans", "start": current_date}, session=s1 )
$items = $client->selectDatabase( 'test', [ 'readConcern' => new \MongoDB\Driver\ReadConcern(\MongoDB\Driver\ReadConcern::MAJORITY), 'writeConcern' => new \MongoDB\Driver\WriteConcern(\MongoDB\Driver\WriteConcern::MAJORITY, 1000), ], )->items; $s1 = $client->startSession( ['causalConsistency' => true], ); $currentDate = new \MongoDB\BSON\UTCDateTime(); $items->updateOne( ['sku' => '111', 'end' => ['$exists' => false]], ['$set' => ['end' => $currentDate]], ['session' => $s1], ); $items->insertOne( ['sku' => '111-nuts', 'name' => 'Pecans', 'start' => $currentDate], ['session' => $s1], );
with client.start_session(causal_consistency=True) as s1: current_date = datetime.datetime.today() items = client.get_database( "test", read_concern=ReadConcern("majority"), write_concern=WriteConcern("majority", wtimeout=1000), ).items items.update_one( {"sku": "111", "end": None}, {"$set": {"end": current_date}}, session=s1 ) items.insert_one( {"sku": "nuts-111", "name": "Pecans", "start": current_date}, session=s1 )
let s1 = client1.startSession(options: ClientSessionOptions(causalConsistency: true)) let currentDate = Date() var dbOptions = MongoDatabaseOptions( readConcern: .majority, writeConcern: try .majority(wtimeoutMS: 1000) ) let items = client1.db("test", options: dbOptions).collection("items") let result1 = items.updateOne( filter: ["sku": "111", "end": .null], update: ["$set": ["end": .datetime(currentDate)]], session: s1 ).flatMap { _ in items.insertOne(["sku": "nuts-111", "name": "Pecans", "start": .datetime(currentDate)], session: s1) }
let s1 = client1.startSession(options: ClientSessionOptions(causalConsistency: true)) let currentDate = Date() var dbOptions = MongoDatabaseOptions( readConcern: .majority, writeConcern: try .majority(wtimeoutMS: 1000) ) let items = client1.db("test", options: dbOptions).collection("items") try items.updateOne( filter: ["sku": "111", "end": .null], update: ["$set": ["end": .datetime(currentDate)]], session: s1 ) try items.insertOne(["sku": "nuts-111", "name": "Pecans", "start": .datetime(currentDate)], session: s1)
別のクライアントが現在の sku
値をすべて読み取る必要がある場合は、クラスター時間と操作時間を他のセッションに合わせて進めて、このクライアントが他のセッションと因果的に一貫していること、および 2 回の書込み後に読み取ることができることを確認できます。
/* Make a new session, session2, and make it causally-consistent * with session1, so that session2 will read session1's writes. */ session2 = mongoc_client_start_session (client, session_opts, &error); if (!session2) { fprintf (stderr, "couldn't start session: %s\n", error.message); goto cleanup; } /* Set the cluster time for session2 to session1's cluster time */ cluster_time = mongoc_client_session_get_cluster_time (session1); mongoc_client_session_advance_cluster_time (session2, cluster_time); /* Set the operation time for session2 to session2's operation time */ mongoc_client_session_get_operation_time (session1, ×tamp, &increment); mongoc_client_session_advance_operation_time (session2, timestamp, increment); /* Run a find on session2, which should now find all writes done * inside of session1 */ find_opts = bson_new (); res = mongoc_client_session_append (session2, find_opts, &error); if (!res) { fprintf (stderr, "couldn't add session to opts: %s\n", error.message); goto cleanup; } find_query = BCON_NEW ("end", BCON_NULL); read_prefs = mongoc_read_prefs_new (MONGOC_READ_SECONDARY); cursor = mongoc_collection_find_with_opts (coll, query, find_opts, read_prefs); while (mongoc_cursor_next (cursor, &result)) { json = bson_as_relaxed_extended_json (result, NULL); fprintf (stdout, "Document: %s\n", json); bson_free (json); } if (mongoc_cursor_error (cursor, &error)) { fprintf (stderr, "cursor failure: %s\n", error.message); goto cleanup; }
using (var session2 = client.StartSession(new ClientSessionOptions { CausalConsistency = true })) { session2.AdvanceClusterTime(session1.ClusterTime); session2.AdvanceOperationTime(session1.OperationTime); var items = client.GetDatabase( "test", new MongoDatabaseSettings { ReadPreference = ReadPreference.Secondary, ReadConcern = ReadConcern.Majority, WriteConcern = new WriteConcern(WriteConcern.WMode.Majority, TimeSpan.FromMilliseconds(1000)) }) .GetCollection<BsonDocument>("items"); var filter = Builders<BsonDocument>.Filter.Eq("end", BsonNull.Value); foreach (var item in items.Find(session2, filter).ToEnumerable()) { // process item } }
// Example 2: Advance the cluster time and the operation time to that of the other session to ensure that // this client is causally consistent with the other session and read after the two writes. ClientSession session2 = client.startSession(ClientSessionOptions.builder().causallyConsistent(true).build()); session2.advanceClusterTime(session1.getClusterTime()); session2.advanceOperationTime(session1.getOperationTime()); items = client.getDatabase("test") .withReadPreference(ReadPreference.secondary()) .withReadConcern(ReadConcern.MAJORITY) .withWriteConcern(WriteConcern.MAJORITY.withWTimeout(1000, TimeUnit.MILLISECONDS)) .getCollection("items"); for (Document item: items.find(session2, eq("end", BsonNull.VALUE))) { System.out.println(item); }
async with await client.start_session(causal_consistency=True) as s2: s2.advance_cluster_time(s1.cluster_time) s2.advance_operation_time(s1.operation_time) items = client.get_database( "test", read_preference=ReadPreference.SECONDARY, read_concern=ReadConcern("majority"), write_concern=WriteConcern("majority", wtimeout=1000), ).items async for item in items.find({"end": None}, session=s2): print(item)
$s2 = $client->startSession( ['causalConsistency' => true], ); $s2->advanceClusterTime($s1->getClusterTime()); $s2->advanceOperationTime($s1->getOperationTime()); $items = $client->selectDatabase( 'test', [ 'readPreference' => new \MongoDB\Driver\ReadPreference(\MongoDB\Driver\ReadPreference::SECONDARY), 'readConcern' => new \MongoDB\Driver\ReadConcern(\MongoDB\Driver\ReadConcern::MAJORITY), 'writeConcern' => new \MongoDB\Driver\WriteConcern(\MongoDB\Driver\WriteConcern::MAJORITY, 1000), ], )->items; $result = $items->find( ['end' => ['$exists' => false]], ['session' => $s2], ); foreach ($result as $item) { var_dump($item); }
with client.start_session(causal_consistency=True) as s2: s2.advance_cluster_time(s1.cluster_time) s2.advance_operation_time(s1.operation_time) items = client.get_database( "test", read_preference=ReadPreference.SECONDARY, read_concern=ReadConcern("majority"), write_concern=WriteConcern("majority", wtimeout=1000), ).items for item in items.find({"end": None}, session=s2): print(item)
let options = ClientSessionOptions(causalConsistency: true) let result2: EventLoopFuture<Void> = client2.withSession(options: options) { s2 in // The cluster and operation times are guaranteed to be non-nil since we already used s1 for operations above. s2.advanceClusterTime(to: s1.clusterTime!) s2.advanceOperationTime(to: s1.operationTime!) dbOptions.readPreference = .secondary let items2 = client2.db("test", options: dbOptions).collection("items") return items2.find(["end": .null], session: s2).flatMap { cursor in cursor.forEach { item in print(item) } } }
try client2.withSession(options: ClientSessionOptions(causalConsistency: true)) { s2 in // The cluster and operation times are guaranteed to be non-nil since we already used s1 for operations above. s2.advanceClusterTime(to: s1.clusterTime!) s2.advanceOperationTime(to: s1.operationTime!) dbOptions.readPreference = .secondary let items2 = client2.db("test", options: dbOptions).collection("items") for item in try items2.find(["end": .null], session: s2) { print(item) } }
制限
メモリ内構造を構築する次の操作は因果整合性はありません。
操作 | ノート |
---|---|
| |
操作に因果整合性があるクライアント セッションに関連付けられている場合はエラーを返します。 | |
操作に因果整合性があるクライアント セッションに関連付けられている場合はエラーを返します。 | |
操作に因果整合性があるクライアント セッションに関連付けられている場合はエラーを返します。 | |
操作に因果整合性があるクライアント セッションに関連付けられている場合はエラーを返します。 | |