AS OF TIMESTAMP
句を使用した履歴データの読み取り
このドキュメントでは、 AS OF TIMESTAMP
句を使用してステイル読み取り機能を実行し、TiDB で履歴データを読み取る方法について説明します。これには、履歴データを保存するための具体的な使用例や戦略も含まれます。
TiDB は、特別なクライアントやドライバーを必要とせず、標準 SQL インターフェイス ( AS OF TIMESTAMP
SQL 句) を介した履歴データの読み取りをサポートします。データが更新または削除された後、この SQL インターフェイスを使用して、更新または削除前の履歴データを読み取ることができます。
ノート:
履歴データを読み取る場合、TiDB は、現在のテーブル構造が異なっていても、古いテーブル構造でデータを返します。
構文
AS OF TIMESTAMP
句は次の 3 つの方法で使用できます。
SELECT ... FROM ... AS OF TIMESTAMP
START TRANSACTION READ ONLY AS OF TIMESTAMP
SET TRANSACTION READ ONLY AS OF TIMESTAMP
正確な時点を指定する場合は、日時値を設定するか、 AS OF TIMESTAMP
句で時刻関数を使用します。日時の形式は「2016-10-08 16:45:26.999」のようなもので、時間の最小単位はミリ秒ですが、ほとんどの場合、「2016」のように日時を指定するには時間単位の秒で十分です。 -10-08 16:45:26」。 NOW(3)
関数を使用して、現在時刻をミリ秒単位で取得することもできます。数秒前のデータを読みたい場合は、 NOW() - INTERVAL 10 SECOND
などの式を使用することをお勧めします。
時間範囲を指定する場合は、句でTIDB_BOUNDED_STALENESS()
関数を使用できます。この関数を使用すると、TiDB は指定された時間範囲内で適切なタイムスタンプを選択します。 「適切」とは、このタイムスタンプより前に開始され、アクセスされたレプリカでコミットされていないトランザクションがないことを意味します。つまり、TiDB はアクセスされたレプリカで読み取り操作を実行でき、読み取り操作はブロックされません。この関数を呼び出すにはTIDB_BOUNDED_STALENESS(t1, t2)
を使用する必要があります。 t1
とt2
は時間範囲の両端であり、日時値または時間関数を使用して指定できます。
AS OF TIMESTAMP
節の例をいくつか示します。
AS OF TIMESTAMP '2016-10-08 16:45:26'
: 2016 年 10 月 8 日の 16:45:26 に保存された最新のデータを読み取るように TiDB に指示します。AS OF TIMESTAMP NOW() - INTERVAL 10 SECOND
: 10 秒前に保存された最新のデータを読み取るように TiDB に指示します。AS OF TIMESTAMP TIDB_BOUNDED_STALENESS('2016-10-08 16:45:26', '2016-10-08 16:45:29')
: 2016 年 10 月 8 日の 16:45:26 から 16:45:29 までの時間範囲内で、できるだけ新しいデータを読み取るように TiDB に指示します。AS OF TIMESTAMP TIDB_BOUNDED_STALENESS(NOW() - INTERVAL 20 SECOND, NOW())
: 20 秒前から現在までの時間範囲内で、できるだけ新しいデータを読み取るように TiDB に指示します。
ノート:
タイムスタンプの指定に加えて、
AS OF TIMESTAMP
句の最も一般的な使用法は、数秒前のデータを読み取ることです。このアプローチを使用する場合は、5 秒より古い履歴データを読み取ることをお勧めします。ステイル読み取りを使用する場合は、TiDB ノードおよび PD ノードに NTP サービスをデプロイする必要があります。これにより、TiDB によって使用される指定されたタイムスタンプが、最新の TSO 割り当て進行状況よりも進んでいる (タイムスタンプが数秒進んでいるなど)、または GC セーフ ポイントのタイムスタンプよりも遅いという状況が回避されます。指定されたタイムスタンプがサービスの範囲を超える場合、TiDB はエラーを返します。
レイテンシーを短縮し、 ステイル読み取りデータの適時性を向上させるために、TiKV
advance-ts-interval
構成項目を変更できます。詳細はステイル読み取りレイテンシーを削減する参照してください。
使用例
このセクションでは、 AS OF TIMESTAMP
句のさまざまな使用方法をいくつかの例とともに説明します。まず、リカバリ用にデータを準備する方法を紹介し、次にSELECT
、 START TRANSACTION READ ONLY AS OF TIMESTAMP
、およびSET TRANSACTION READ ONLY AS OF TIMESTAMP
のAS OF TIMESTAMP
それぞれ使用する方法を示します。
データサンプルの準備
リカバリ用のデータを準備するには、最初にテーブルを作成し、いくつかのデータ行を挿入します。
create table t (c int);
Query OK, 0 rows affected (0.01 sec)
insert into t values (1), (2), (3);
Query OK, 3 rows affected (0.00 sec)
テーブル内のデータをビュー。
select * from t;
+------+
| c |
+------+
| 1 |
| 2 |
| 3 |
+------+
3 rows in set (0.00 sec)
現在の時刻をビュー:
select now();
+---------------------+
| now() |
+---------------------+
| 2021-05-26 16:45:26 |
+---------------------+
1 row in set (0.00 sec)
データを連続して更新します。
update t set c=22 where c=2;
Query OK, 1 row affected (0.00 sec)
行のデータが更新されたことを確認します。
select * from t;
+------+
| c |
+------+
| 1 |
| 22 |
| 3 |
+------+
3 rows in set (0.00 sec)
SELECT
ステートメントを使用して履歴データを読み取る
SELECT ... FROM ... AS OF TIMESTAMP
ステートメントを使用すると、過去のある時点からデータを読み取ることができます。
select * from t as of timestamp '2021-05-26 16:45:26';
+------+
| c |
+------+
| 1 |
| 2 |
| 3 |
+------+
3 rows in set (0.00 sec)
ノート:
1 つの
SELECT
ステートメントを使用して複数のテーブルを読み取る場合は、TIMESTAMP EXPRESSION の形式が一貫していることを確認する必要があります。たとえば、select * from t as of timestamp NOW() - INTERVAL 2 SECOND, c as of timestamp NOW() - INTERVAL 2 SECOND;
。さらに、SELECT
ステートメントで関連テーブルのAS OF
情報を指定する必要があります。それ以外の場合、SELECT
ステートメントはデフォルトで最新のデータを読み取ります。
START TRANSACTION READ ONLY AS OF TIMESTAMP
ステートメントを使用して履歴データを読み取る
START TRANSACTION READ ONLY AS OF TIMESTAMP
ステートメントを使用すると、過去の時点に基づいて読み取り専用トランザクションを開始できます。トランザクションは、指定された時点の履歴データを読み取ります。
start transaction read only as of timestamp '2021-05-26 16:45:26';
Query OK, 0 rows affected (0.00 sec)
select * from t;
+------+
| c |
+------+
| 1 |
| 2 |
| 3 |
+------+
3 rows in set (0.00 sec)
commit;
Query OK, 0 rows affected (0.00 sec)
トランザクションがコミットされた後、最新のデータを読み取ることができます。
select * from t;
+------+
| c |
+------+
| 1 |
| 22 |
| 3 |
+------+
3 rows in set (0.00 sec)
ノート:
ステートメント
START TRANSACTION READ ONLY AS OF TIMESTAMP
でトランザクションを開始すると、それは読み取り専用トランザクションになります。このトランザクションでは書き込み操作が拒否されます。
SET TRANSACTION READ ONLY AS OF TIMESTAMP
ステートメントを使用して履歴データを読み取る
SET TRANSACTION READ ONLY AS OF TIMESTAMP
ステートメントを使用すると、過去の指定された時点に基づいて次のトランザクションを読み取り専用トランザクションとして設定できます。トランザクションは、指定された時点の履歴データを読み取ります。
set transaction read only as of timestamp '2021-05-26 16:45:26';
Query OK, 0 rows affected (0.00 sec)
begin;
Query OK, 0 rows affected (0.00 sec)
select * from t;
+------+
| c |
+------+
| 1 |
| 2 |
| 3 |
+------+
3 rows in set (0.00 sec)
commit;
Query OK, 0 rows affected (0.00 sec)
トランザクションがコミットされた後、最新のデータを読み取ることができます。
select * from t;
+------+
| c |
+------+
| 1 |
| 22 |
| 3 |
+------+
3 rows in set (0.00 sec)
ノート:
ステートメント
SET TRANSACTION READ ONLY AS OF TIMESTAMP
でトランザクションを開始すると、それは読み取り専用トランザクションになります。このトランザクションでは書き込み操作が拒否されます。