クエリに対応した設計

Table service ソリューションでは、読み取り、書き込み、またはその両方の負荷が高くなることがあります。 この記事では、読み取り操作を効率的に行える Table service を設計する際に注意する必要のある事柄を中心に取り上げます。 通常は、読み取り操作を効率的にサポートする設計は、書き込み操作についても効率が高くなります。 ただし、書き込み操作をサポートする設計には、さらに考慮すべき事柄があります。「データの変更に対応した設計」の記事を参照してください。

データを効率的に読み取ることのできる Table service ソリューションを設計する際には、まず "必要なデータを Table service から取得するには、アプリケーションでどのようなクエリを実行する必要があるか" を考えてみてください。

Note

Table service を使用する場合は、事前に正しく設計することが重要です。後で設計を変更するのは難しいだけでなく、コストも高くなるためです。 たとえば、リレーショナル データベースなら、既存のデータベースにインデックスを追加するだけでパフォーマンスの問題に対処できるのが普通ですが、Table service ではそうはいきません。

このセクションでは、クエリに対応したテーブルを設計する際に対処する必要のある主要な問題を中心に取り上げます。 このセクションで取り上げるトピックは次のとおりです。

選択した PartitionKey と RowKey がクエリのパフォーマンスに与える影響

次の例は、以下の構造を持った複数の従業員エンティティの格納を想定しています (明確にするためほとんどの例で タイムスタンプ プロパティを省略)。

列名 データの種類
PartitionKey (部門名) String
RowKey (従業員 ID) String
FirstName String
LastName String
Age Integer
EmailAddress String

Azure Table Storage の概要」の記事では、クエリの設計に直接影響を与える Azure Table service の主な機能の一部について説明します。 ここから、Table service のクエリを設計する際には、次のような一般的なガイドラインが考えられます。 以下の例で使用しているフィルター構文は、Table service REST API の構文です。詳細については、「Query Entities (エンティティの照会)」をご覧ください。

  • ポイント クエリ は、最も効率的な検索です。大量の参照または短い待機時間が求められる参照に使用することが推奨されます。 このようなクエリでは、PartitionKeyRowKey 値の両方を指定することでインデックスを使用し、個別のエンティティを非常に効率よく検索することができます。 例: $filter (PartitionKey eq 'Sales') = および (RowKey eq '2')
  • 2 番目に良い方法は、PartitionKey を使用する "範囲クエリ" と、RowKey 値の範囲にフィルターをかけ、1 つ以上のエンティティを返すというものです。 PartitionKey 値は特定のパーティションを識別し、RowKey 値はそのパーティション内のエンティティのサブセットを識別します。 例: $filter=PartitionKey eq 'Sales' および RowKey ge 'S' および RowKey lt 'T'
  • 3 番目に良い方法は、PartitionKey を使用し、他のキーを持たないプロパティにフィルターをかけ、1 つ以上のエンティティを返すことが可能な "パーティション スキャン" です。 PartitionKey 値は特定のパーティションを識別し、プロパティ値はそのパーティション内のエンティティのサブセットを選択します。 例: $filter = PartitionKey eq '販売'、および LastName eq 'Smith'
  • "テーブル スキャン" に PartitionKey は含まれません。また、一致するエンティティのテーブルを構成するパーティションのすべてを検索するため、非常に非効率的です。 フィルターが RowKey を使用する / しないにかかわらず、テーブルのスキャンが実行されます。 例: $filter = LastName eq 'Jones'
  • クエリは複数のエンティティを PartitionKeyRowKey の順序で並べ替えて返します。 クライアント内でエンティティを再度並べ替えるのを防ぐため、最も一般的な並べ替え順序を定義する RowKey を選択します。

"or" を使用して RowKey 値に基づいてフィルターを指定した場合はパーティション スキャンが行われます。範囲クエリとしては扱われません。 そのため、$filter = PartitionKey eq 'Sales' and (RowKey eq '121' or RowKey eq '322') のようなフィルターを使用するクエリの使用は避ける必要があります。

ストレージ クライアント ライブラリを使って効率的なクエリを実行するクライアント側コードの例については、

同じテーブルに格納された複数の種類のエンティティを処理できるクライアント側コードの例については、

適切な PartitionKey の選択

PartitionKey の選択においては、(一貫性の確保に必要となる) エンティティ グループ トランザクションの使用を可能にするため、複数のパーティションでエンティティを配布するための要件に対し、(スケーラブルなソリューションの確保のため) 必要性のバランスを取ることが求められます。

極端な例として、すべてのエンティティを 1 つのパーティションに格納することも可能ですが、そうするとソリューションのスケーラビリティが制限され、Table サービスで要求を負荷分散できなくなる可能性があります。 逆に、パーティションごとに 1 つのエンティティを格納することもできます。このようにすると、スケーラビリティが高まり、Table サービスで要求を負荷分散できるようになるものの、エンティティ グループ トランザクションは使用できなくなります。

理想的な PartitionKey とは、効率的なクエリを使用することができ、スケーラブルなソリューションを確保できるパーティションを持つものを指します。 通常、エンティティには、十分な数のパーティションにエンティティを分散できるだけのプロパティがあります。

Note

たとえば、ユーザーまたは従業員に関する情報を格納するシステムでは、ユーザー ID を PartitionKey として使うと便利です。 パーティション分割キーとして所定の UserID を使用する複数のエンティティを使用できます。 あるユーザーに関するデータを保存するそれぞれのエンティティは 1 つのパーティションにグループ化されるため、高い拡張性を確保しながら、エンティティ グループ トランザクションを介してこれらのエンティティにアクセスできます。

選択した PartitionKey についてエンティティの挿入、更新、削除方法に関連する追加の検討事項があります。 詳細については、データ変更のためのテーブルの設計に関するセクションを参照してください。

クエリを Table service 向けに最適化する

Table service は 1 つのクラスター化インデックス内の PartitionKeyRowKey 値を使用して自動的にインデックスを作成します。そのため、ポイント クエリの使用が最も効果的です。 ただし、PartitionKeyRowKey のクラスター化インデックス以外にはインデックスはありません。

多くの設計で、複数の条件に基づいてエンティティを参照できるようにするという要件を満たす必要があります。 たとえば、電子メール、従業員 ID、姓に基づいて従業員エンティティを特定する場合などです。 「テーブルの設計パターン」で説明されているパターンでは、このような要件の種類に対応しています。また、Table service がセカンダリ インデックスを提供しないことへの対処方法について説明します。

Table service でデータを並べ替える

Table service は、PartitionKey 次に RowKey に基づいた昇順で並べ替え、エンティティを返します。 これらのキーは文字列値であり、数値を正しく並べ替えるには、固定長の値に変換し、ゼロ パディングを施す必要があります。 たとえば、従業員 ID 値を整数値の RowKey として使用する場合、従業員 ID を 123 から 00000123 に変換する必要があります。

さまざまな順序 (名前、入社日など) で並べ替えられたデータを使う必要のあるアプリケーションは多数あります。 次のパターンでは、エンティティの並べ替え順序を代替する方法に対応しています。

次のステップ