SQLite、複数クライアントからの同時書き込みを可能にする「BEGIN CONCURRENT」文を実装へ

今回は「SQLite、複数クライアントからの同時書き込みを可能にする「BEGIN CONCURRENT」文を実装へ」についてご紹介します。

関連ワード (実装、慎重、操作等) についても参考にしながら、ぜひ本記事について議論していってくださいね。

本記事は、Publickey様で掲載されている内容を参考にしておりますので、より詳しく内容を知りたい方は、ページ下の元記事リンクより参照ください。


SQLiteの開発チームは、複数クライアントからの同時書き込みを可能にするBEGIN CONCURRENT文を実装していることを明らかにしました。

これまでSQLiteでは書き込みの同時実行はできず、つねに1つのクライアントだけが書き込み可能でした。

fig

同時書き込み処理は、データベースのジャーナルモードが「wal」(Write-Ahead-log)もしくはwalを改良した「wal2」で、BEGIN CONCURRENT文を実行した場合に可能となります。

どのように同時書き込み処理が行われるのかについては、上記のWebページの説明を引用しましょう。

ロックが延期されることで同時書き込みが可能に

まず、書き込み時のロックがCOMMITまで延期されることで同時書き込みが実現されると説明されています。

When a write-transaction is opened with “BEGIN CONCURRENT”, actually locking the database is deferred until a COMMIT is executed. This means that any number of transactions started with BEGIN CONCURRENT may proceed concurrently. The system uses optimistic page-level-locking to prevent conflicting concurrent transactions from being committed.

BEGIN CONCURRENTによって書き込みトランザクションが開始されると、COMMITが実行されるまでデータベースのロックは延期されます。これによりBEGIN CONCURRENTで開始されたトランザクションは複数同時に実行できるようになります。
システムは楽観的ページレベルロッキングにより競合している同時トランザクションがコミットされるのを防ぎます。

COMMITされた時点で、トランザクションの衝突が発生していないかがチェックされることになります。

When a BEGIN CONCURRENT transaction is committed, the system checks whether or not any of the database pages that the transaction has read have been modified since the BEGIN CONCURRENT was opened. In other words – it asks if the transaction being committed operates on a different set of data than all other concurrently executing transactions. If the answer is “yes, this transaction did not read or modify any data modified by any concurrent transaction”, then the transaction is committed as normal.

BEGIN CONCURRENTトランザクションがコミットされると、システムはBEGIN CONCURRENTがオープンされてから、トランザクションが読み込んだデータベースのページが変更されたかどうかをチェックします。言い換えると、コミットされるトランザクションが、他のすべての同時実行トランザクションとは異なるデータ集合を操作しているかどうかを確認します。
その答えが 「はい、このトランザクションは同時実行中のトランザクションによって変更されたデータを読み込んだり変更したりしませんでした」であれば、そのトランザクションは通常通りコミットされます。

もしトランザクションの衝突が発生していた場合、ロールバックのみが可能です。

Otherwise, if the transaction does conflict, it cannot be committed and an SQLITE_BUSY_SNAPSHOT error is returned. At this point, all the client can do is ROLLBACK the transaction.
If SQLITE_BUSY_SNAPSHOT is returned, messages are output via the sqlite3_log mechanism indicating the page and table or index on which the conflict occurred. This can be useful when optimizing concurrency.

そうでない場合、トランザクションが衝突していればコミットできず、SQLITE_BUSY_SNAPSHOTエラーが返されます。この時点でクライアントにできることは、トランザクションをロールバックすることだけです。
SQLITE_BUSY_SNAPSHOTが返された場合、sqlite3_logメカニズムを通じて、競合が発生したページとテーブルまたはインデックスを示すメッセージが出力されます。これは同時実行を最適化する際に有用です。

このCOMMIT処理はつねに1つずつ直列化されて行われます。

In order to serialize COMMIT processing, SQLite takes a lock on the database as part of each COMMIT command and releases it before returning. At most one writer may hold this lock at any one time. If a writer cannot obtain the lock, it uses SQLite’s busy-handler to pause and retry for a while:

COMMIT処理を直列化するために、SQLiteは各COMMITコマンドの一部としてデータベースに対するロックを取り、戻る前にロックを解放します。一度にこのロックを保持できるライターは1つまでです。ライターがロックを取得できない場合、SQLiteのbusy-handlerを使用して一時停止し、しばらく再試行します:

同時書き込みを最大化するには?

アプリケーション開発者としてはこの同時書き込み機能を活用するために、できるだけトランザクションの衝突を起こさないように配慮するべきでしょう。

SQLiteでは、各テーブルと各インデックスは個別のb-treeとして格納され、それぞれが個別のデータベースページの集合に分散されているため以下の2つが言えると説明されています。

1)異なるテーブル・セットに書き込む2つのトランザクションが衝突することはない。

2)同じテーブルまたはインデックスに書き込む2つのトランザクションが衝突するのは、キー(主キーまたはインデックス行)の値がかなり近い場合だけである。

つまりテーブルが異なればページも異なるため、衝突することはなく、同じテーブルであってもキーの値が遠ければページが異なる可能性が高い、ということのようです。

SQLite 3での同時書き込みによるアプリケーションの高速化の恩恵を受けるために、今後はこれらを可能な範囲で配慮したデータベースのスキーマや格納する値などを慎重に設計するとよさそうです。

COMMENTS


Recommended

TITLE
CATEGORY
DATE
DX推進のための環境整備–推進環境の成熟度を高める企業内変革
IT関連
2022-08-18 19:26
LG化学がEV用バッテリー生産拡大へ向け2025年までに5770億円を投資
モビリティ
2021-07-16 10:31
COCOA、修正版配布も課題残る 「Androidは1日1回アプリの再起動を」「iOSは最新の14に」
企業・業界動向
2021-02-19 22:48
動く腕にぴったり貼り付いて映像を投影 東工大が開発 :Innovative Tech
トップニュース
2021-04-07 13:27
日本人の現金使用額は中国人の7倍 銀聯が日中韓のキャッシュレス動向を調査
社会とIT
2021-08-17 05:35
決済データで事業運営を最適化–グローバル企業のCXを支えるAdyen
IT関連
2023-06-30 00:16
IBMと東大ら5大学、日米韓で4万人の学生に量子教育を推進
IT関連
2023-12-16 18:22
富士フイルムとIBM、世界最大記録容量のテープストレージを開発
IT関連
2023-08-31 16:17
「2030年、ロボットに仕事を奪われる現役世代は19%」 理系の大学教授ら112人が予測
ロボット・AI
2021-08-13 10:29
ヤマダHD、全社で業務のデジタル化を推進–年300時間程度の作業時間を削減
IT関連
2022-06-11 13:33
250万ユーザーのアフリカの農家向けSNS「Wefarm」が独立農家のネットワーク拡大に向け約12億円追加調達
ネットサービス
2021-05-11 03:59
OpenAI、会話を記憶できる機能を「ChatGPT Plus」ユーザーに提供開始
IT関連
2024-05-01 13:27
エクイニクス、東京で2番目のハイパースケーラー向けデータセンター開設
IT関連
2023-07-05 23:43
顔交換アプリ「Reface」でユーザーがアップロードした画像の顔入れ替え&アニメーション化が可能に
ソフトウェア
2021-05-19 22:38