メインコンテンツまでスキップ
バージョン: 4.x

ベストプラクティス

非同期マテリアライズドビューの使用原則

  • 適時性の考慮: 非同期マテリアライズドビューは通常、データの適時性が重要でないシナリオで使用され、一般的にT+1データです。高い適時性が必要な場合は、同期マテリアライズドビューの使用を検討してください。

  • 高速化効果と一貫性の考慮: クエリ高速化のシナリオでは、マテリアライズドビューを作成する際、DBAは共通のクエリSQLパターンをグループ化し、グループ間の重複を最小限に抑えることを目指すべきです。SQLパターンのグループ化が明確であるほど、マテリアライズドビューの構築品質は高くなります。1つのクエリで複数のマテリアライズドビューを使用することができ、1つのマテリアライズドビューが複数のクエリで使用されることもあります。マテリアライズドビューの構築には、レスポンス時間(高速化効果)、構築コスト、およびデータ一貫性要件の包括的な考慮が必要です。

  • マテリアライズドビューの定義と構築コストの考慮:

    • マテリアライズドビューの定義が元のクエリに近いほど、クエリ高速化効果は良くなりますが、マテリアライゼーションの汎用性と再利用性は低くなり、構築コストが高くなります。

    • マテリアライズドビューの定義がより汎用的である場合(例:WHERE条件がなく、より多くの集約ディメンション)、クエリ高速化効果は低くなりますが、マテリアライゼーションの汎用性と再利用性は向上し、構築コストが低くなります。

Note
  • マテリアライズドビュー数の制御: マテリアライズドビューは多ければ良いというものではありません。マテリアライズドビューの構築とリフレッシュにはリソースが必要です。マテリアライズドビューは透過的リライトに参加し、CBOコストモデルが最適なマテリアライズドビューを選択するのに時間が必要です。理論的には、マテリアライズドビューが多いほど、透過的リライト時間は長くなります。

  • マテリアライズドビューの使用状況を定期的にチェック: 使用されていない場合は、適時削除する必要があります。

  • ベースTableのデータ更新頻度: マテリアライズドビューのベースTableデータが頻繁に更新される場合、マテリアライズドビューの使用は適さない可能性があります。これにより、マテリアライズドビューが頻繁に無効になり、透過的リライト(直接クエリ)で使用できなくなるためです。このようなマテリアライズドビューを透過的リライトで使用する必要がある場合、クエリされるデータに一定の適時性遅延を許可し、grace_periodを設定することができます。grace_periodの適用に関する詳細な説明を参照してください。

マテリアライズドビューのリフレッシュ方法選択の原則

以下の条件が満たされる場合、パーティション化されたマテリアライズドビューの作成を推奨します:

  • マテリアライズドビューのベースTableのデータ量が大きく、ベースTableがパーティションTableである。

  • マテリアライズドビューで使用されるTableのうち、パーティションTable以外は頻繁に変更されない。

  • マテリアライズドビューの定義SQLとパーティションフィールドが、パーティション派生の要件を満たす、つまりパーティション増分更新の要件を満たす。詳細な要件はCREATE-ASYNC-MATERIALIZED-VIEWで確認できます。

  • マテリアライズドビューのパーティション数が大きくない。パーティションが多すぎると、パーティションマテリアライズドビューの構築時間が過度に長くなります。

マテリアライズドビューの一部のパーティションが無効になった場合、透過的リライトはマテリアライズドビューの有効なパーティションとベースTableのUNION ALLを使用してデータを返すことができます。

パーティション化されたマテリアライズドビューが構築できない場合は、フルリフレッシュマテリアライズドビューの選択を検討できます。

パーティション化されたマテリアライズドビューの一般的な使用法

マテリアライズドビューのベースTableのデータ量が大きく、ベースTableがパーティションTableである場合、マテリアライズドビューの定義SQLとパーティションフィールドがパーティション派生の要件を満たしていれば、このシナリオはパーティション化されたマテリアライズドビューの構築に適しています。パーティション派生の詳細な要件については、CREATE-ASYNC-MATERIALIZED-VIEWとAsync Materialized View FAQ構築質問12を参照してください。

マテリアライズドビューのパーティションは、ベースTableのパーティションマッピングに従って作成され、一般的にベースTableのパーティションと1:1または1:nの関係があります。

  • ベースTableのパーティションでデータ変更が発生した場合(パーティションの追加や削除など)、マテリアライズドビューの対応するパーティションも無効になります。無効なパーティションは透過的リライトに使用できませんが、直接クエリすることはできます。透過的リライトがマテリアライズドビューのパーティションデータが無効であることを発見した場合、無効なパーティションはベースTableとの結合によって処理され、クエリに応答します。

    マテリアライズドビューのパーティション状態を確認するコマンドについては、マテリアライズドビューの状態表示を参照してください。主にshow partitions from mv_nameコマンドを使用します。

  • マテリアライズドビューで参照される非パーティションTableでデータ変更が発生した場合、マテリアライズドビューのすべてのパーティションが無効になり、マテリアライズドビューが透過的リライトで使用できなくなります。REFRESH MATERIALIZED VIEW mv1 AUTO;コマンドを使用して、マテリアライズドビューのすべてのパーティションデータをリフレッシュする必要があります。このコマンドは、データが変更されたマテリアライズドビューのすべてのパーティションのリフレッシュを試行します。

    したがって、一般的には、頻繁に変更されるデータをパーティション化されたマテリアライズドビューで参照されるパーティションTableに配置し、変更頻度の低いディメンションTableを非参照パーティションTableの位置に配置することを推奨します。

  • マテリアライズドビューで参照される非パーティションTableでデータ変更が発生し、非パーティションTableデータが修正なしで追加のみの場合、マテリアライズドビュー作成時に属性excluded_trigger_tables = 'non_partition_table_name1,non_partition_table_name2'を指定できます。これにより、非パーティションTableでのデータ変更はマテリアライズドビューのすべてのパーティションを無効にせず、次回のリフレッシュではパーティションTableに対応するマテリアライズドビューの無効なパーティションのみがリフレッシュされます。

パーティション化されたマテリアライズドビューの透過的リライトはパーティション粒度で行われます。マテリアライズドビューの一部のパーティションが無効になっても、マテリアライズドビューは透過的リライトで使用できます。ただし、1つのパーティションのみをクエリし、そのパーティションのマテリアライズドビューのデータが無効な場合、マテリアライズドビューは透過的リライトで使用できません。

例えば:

CREATE TABLE IF NOT EXISTS lineitem (
l_orderkey INTEGER NOT NULL,
l_partkey INTEGER NOT NULL,
l_suppkey INTEGER NOT NULL,
l_linenumber INTEGER NOT NULL,
l_ordertime DATETIME NOT NULL,
l_quantity DECIMALV3(15, 2) NOT NULL,
l_extendedprice DECIMALV3(15, 2) NOT NULL,
l_discount DECIMALV3(15, 2) NOT NULL,
l_tax DECIMALV3(15, 2) NOT NULL,
l_returnflag CHAR(1) NOT NULL,
l_linestatus CHAR(1) NOT NULL,
l_shipdate DATE NOT NULL,
l_commitdate DATE NOT NULL,
l_receiptdate DATE NOT NULL,
l_shipinstruct CHAR(25) NOT NULL,
l_shipmode CHAR(10) NOT NULL,
l_comment VARCHAR(44) NOT NULL
) DUPLICATE KEY(
l_orderkey, l_partkey, l_suppkey,
l_linenumber
) PARTITION BY RANGE(l_ordertime) (
FROM
('2024-05-01') TO ('2024-06-30') INTERVAL 1 DAY
)
DISTRIBUTED BY HASH(l_orderkey) BUCKETS 3;

INSERT INTO lineitem VALUES
(1, 2, 3, 4, '2024-05-01 01:45:05', 5.5, 6.5, 0.1, 8.5, 'o', 'k', '2024-05-01', '2024-05-01', '2024-05-01', 'a', 'b', 'yyyyyyyyy'),
(1, 2, 3, 4, '2024-05-15 02:35:05', 5.5, 6.5, 0.15, 8.5, 'o', 'k', '2024-05-15', '2024-05-15', '2024-05-15', 'a', 'b', 'yyyyyyyyy'),
(2, 2, 3, 5, '2024-05-25 08:30:06', 5.5, 6.5, 0.2, 8.5, 'o', 'k', '2024-05-25', '2024-05-25', '2024-05-25', 'a', 'b', 'yyyyyyyyy'),
(3, 4, 3, 6, '2024-06-02 09:25:07', 5.5, 6.5, 0.3, 8.5, 'o', 'k', '2024-06-02', '2024-06-02', '2024-06-02', 'a', 'b', 'yyyyyyyyy'),
(4, 4, 3, 7, '2024-06-15 13:20:09', 5.5, 6.5, 0, 8.5, 'o', 'k', '2024-06-15', '2024-06-15', '2024-06-15', 'a', 'b', 'yyyyyyyyy'),
(5, 5, 6, 8, '2024-06-25 15:15:36', 5.5, 6.5, 0.12, 8.5, 'o', 'k', '2024-06-25', '2024-06-25', '2024-06-25', 'a', 'b', 'yyyyyyyyy'),
(5, 5, 6, 9, '2024-06-29 21:10:52', 5.5, 6.5, 0.1, 8.5, 'o', 'k', '2024-06-30', '2024-06-30', '2024-06-30', 'a', 'b', 'yyyyyyyyy'),
(5, 6, 5, 10, '2024-06-03 22:05:50', 7.5, 8.5, 0.1, 10.5, 'k', 'o', '2024-06-03', '2024-06-03', '2024-06-03', 'c', 'd', 'xxxxxxxxx');

CREATE TABLE IF NOT EXISTS partsupp (
ps_partkey INTEGER NOT NULL,
ps_suppkey INTEGER NOT NULL,
ps_availqty INTEGER NOT NULL,
ps_supplycost DECIMALV3(15, 2) NOT NULL,
ps_comment VARCHAR(199) NOT NULL
)
DUPLICATE KEY(ps_partkey, ps_suppkey)
DISTRIBUTED BY HASH(ps_partkey) BUCKETS 3;


INSERT INTO partsupp VALUES
(2, 3, 9, 10.01, 'supply1'),
(4, 3, 9, 10.01, 'supply2'),
(5, 6, 9, 10.01, 'supply3'),
(6, 5, 10, 11.01, 'supply4');

この例では、ordersTableのo_ordertimeフィールドがパーティションフィールドであり、型はDATETIMEで、日単位でパーティション化されています。 メインクエリは「日」の粒度に基づいています:

SELECT 
l_linestatus,
sum(
l_extendedprice * (1 - l_discount)
) AS revenue,
ps_partkey
FROM
lineitem
LEFT JOIN partsupp ON l_partkey = ps_partkey
and l_suppkey = ps_suppkey
WHERE
date_trunc(l_ordertime, 'day') <= DATE '2024-05-25'
AND date_trunc(l_ordertime, 'day') >= DATE '2024-05-05'
GROUP BY
l_linestatus,
ps_partkey;

マテリアライズドビューで毎回あまりに多くのパーティションをリフレッシュすることを避けるために、パーティション粒度をベースTableのordersと一致させ、同様に"day"でパーティショニングすることができます。

マテリアライズドビューの定義SQLは"day"粒度を使用し、"day"でデータを集約できます:

CREATE MATERIALIZED VIEW rollup_partition_mv 
BUILD IMMEDIATE REFRESH AUTO ON MANUAL
partition by(order_date)
DISTRIBUTED BY RANDOM BUCKETS 2
AS
SELECT
l_linestatus,
sum(
l_extendedprice * (1 - l_discount)
) AS revenue,
ps_partkey,
date_trunc(l_ordertime, 'day') as order_date
FROM
lineitem
LEFT JOIN partsupp ON l_partkey = ps_partkey
and l_suppkey = ps_suppkey
GROUP BY
l_linestatus,
ps_partkey,
date_trunc(l_ordertime, 'day');

UNION ALLを使用したパーティション化マテリアライズドビューの作成

現在、Dorisにはパーティション化マテリアライズドビューの定義にUNION ALL句を含めることができないという制限があります。 UNION ALLを含むマテリアライズドビューを作成するには、次のアプローチを使用できます:UNION ALLの各入力部分に対して、パーティション化マテリアライズドビューの作成を試行し、その後UNION ALL結果セット全体に対して通常のビューを作成します。

例: 以下のマテリアライズドビュー定義にはUNION ALL句が含まれており、パーティション化マテリアライズドビューを直接作成することはできません。


SELECT
l_linestatus,
sum(
l_extendedprice * (1 - l_discount)
) AS revenue,
ps_partkey,
date_trunc(l_ordertime, 'day') as order_date
FROM
lineitem
LEFT JOIN partsupp ON l_partkey = ps_partkey
and l_suppkey = ps_suppkey
GROUP BY
l_linestatus,
ps_partkey,
date_trunc(l_ordertime, 'day')
UNION ALL
SELECT
l_linestatus,
l_extendedprice,
ps_partkey,
date_trunc(l_ordertime, 'day') as order_date
FROM
lineitem
LEFT JOIN partsupp ON l_partkey = ps_partkey
and l_suppkey = ps_suppkey;

上記のSQL文を2つの部分に分割し、2つのパーティション化されたマテリアライズドビューを個別に作成することができます。


CREATE MATERIALIZED VIEW union_sub_mv1
BUILD IMMEDIATE REFRESH AUTO ON MANUAL
partition by(order_date)
DISTRIBUTED BY RANDOM BUCKETS 2
AS
SELECT
l_linestatus,
sum(
l_extendedprice * (1 - l_discount)
) AS revenue,
ps_partkey,
date_trunc(l_ordertime, 'day') as order_date
FROM
lineitem
LEFT JOIN partsupp ON l_partkey = ps_partkey
and l_suppkey = ps_suppkey
GROUP BY
l_linestatus,
ps_partkey,
date_trunc(l_ordertime, 'day');
CREATE MATERIALIZED VIEW union_sub_mv2
BUILD IMMEDIATE REFRESH AUTO ON MANUAL
partition by(order_date)
DISTRIBUTED BY RANDOM BUCKETS 2
AS
SELECT
l_linestatus,
l_extendedprice,
ps_partkey,
date_trunc(l_ordertime, 'day') as order_date
FROM
lineitem
LEFT JOIN partsupp ON l_partkey = ps_partkey
and l_suppkey = ps_suppkey;

次に、2つの分割されたマテリアライズドビューの結果セットに対してUNION ALLを実行する通常のビューを作成します。 このビュー(union_all_view)は外部に公開することができます。

CREATE VIEW union_all_view
AS
SELECT *
FROM
union_sub_mv1
UNION ALL
SELECT *
FROM
union_sub_mv2;

最新のパーティションデータのみを保持するパーティション化マテリアライズドビュー

Note

この機能はApache Dorisバージョン2.1.1からサポートされています。

マテリアライズドビューは最新のパーティションからのデータのみを保持するように設定でき、各リフレッシュ時に期限切れのパーティションデータを自動的に削除します。 これは、マテリアライズドビューに以下のプロパティを設定することで実現できます: partition_sync_limit、partition_sync_time_unit、およびpartition_sync_date_format。

partition_sync_limit:ベースTableのパーティションフィールドが時間ベースの場合、このプロパティはベースTableパーティションの同期範囲を設定し、partition_sync_time_unitと組み合わせて動作します。例えば、partition_sync_time_unitをDAYとして3に設定すると、ベースTableの過去3日間のパーティションとデータのみが同期されます。

partition_sync_time_unit:パーティションリフレッシュの時間単位で、DAY/MONTH/YEAR をサポートします(デフォルトはDAY)。

partition_date_format:ベースTableのパーティションフィールドが文字列型の場合、partition_sync_limit機能を使用したい場合にこのプロパティで日付フォーマットを設定します。

例: 以下で定義されたマテリアライズドビューは、過去3日間のデータのみを保持します。最近の3日間にデータがない場合、このマテリアライズドビューを直接クエリしても結果は返されません。

CREATE MATERIALIZED VIEW latest_partition_mv
BUILD IMMEDIATE REFRESH AUTO ON MANUAL
PARTITION BY(order_date)
DISTRIBUTED BY RANDOM BUCKETS 2
PROPERTIES (
"partition_sync_limit" = "3",
"partition_sync_time_unit" = "DAY",
"partition_date_format" = "yyyy-MM-dd"
)
AS
SELECT
l_linestatus,
sum(
l_extendedprice * (1 - l_discount)
) AS revenue,
ps_partkey,
date_trunc(l_ordertime, 'day') as order_date
FROM
lineitem
LEFT JOIN partsupp ON l_partkey = ps_partkey
AND l_suppkey = ps_suppkey
GROUP BY
l_linestatus,
ps_partkey,
date_trunc(l_ordertime, 'day');

マテリアライズドビューを使用してクエリを高速化する方法

クエリの高速化にマテリアライズドビューを使用するには、まずプロファイルファイルを確認して、クエリで最も時間を消費する操作を見つけます。これは通常、Join、Aggregate、Filter、またはCalculated Expressionsに現れます。

Join、Aggregate、Filter、およびCalculated Expressionsについては、マテリアライズドビューを構築することでクエリの高速化に役立ちます。クエリ内のJoin操作が大量のコンピューティングリソースを消費する一方で、Aggregateが相対的に少ないリソースしか消費しない場合、Join操作を対象としたマテリアライズドビューを構築できます。

次に、これら4つの操作に対するマテリアライズドビューの構築方法について詳しく説明します:

  1. Joinの場合

    クエリで使用される共通のTable結合パターンを抽出して、マテリアライズドビューを構築できます。透過的な書き換えがこのマテリアライズドビューを使用する場合、Join計算を節約できます。より汎用的なJoinマテリアライズドビューを作成するために、クエリからFilterを削除します。

  2. Aggregateの場合

    マテリアライズドビューを構築する際は、低カーディナリティフィールドをディメンションとして使用することを推奨します。ディメンションが関連している場合、集約後の数を可能な限り削減できます。

    例えば、Tablet1において、元のTableに1,000,000のレコードがあり、SQLクエリにgroup by a, b, cがある場合を考えます。a、b、cのカーディナリティがそれぞれ100、50、15である場合、集約されたデータは約75,000になり、このマテリアライズドビューが効果的であることを示します。a、b、cが相関している場合、集約データの量はさらに削減されます。

    a、b、cが高いカーディナリティを持つ場合、集約データが急速に拡大します。集約データが元のTableデータよりも多い場合、このシナリオはマテリアライズドビューの構築に適さない可能性があります。例えば、cのカーディナリティが3,500の場合、集約データは約17,000,000になり、元のTableデータよりもはるかに大きくなるため、このようなマテリアライズドビューを構築することによるパフォーマンス向上の効果は低くなります。

    マテリアライズドビューの集約粒度はクエリよりも細かくする必要があります。つまり、マテリアライズドビューの集約ディメンションは、クエリに必要なデータを提供するために、クエリの集約ディメンションを含む必要があります。クエリはGroup Byを記述しない場合があり、同様に、マテリアライズドビューの集約関数はクエリの集約関数を含む必要があります。

    aggregate クエリ加速を例に取ります:

    Query 1:

    SELECT 
    l_linestatus,
    sum(
    l_extendedprice * (1 - l_discount)
    ) AS revenue,
    o_shippriority
    FROM
    orders
    LEFT JOIN lineitem ON l_orderkey = o_orderkey
    WHERE
    o_orderdate <= DATE '2024-06-30'
    AND o_orderdate >= DATE '2024-05-01'
    GROUP BY
    l_linestatus,
    o_shippriority,
    l_partkey;

Query 2:

```sql
SELECT
l_linestatus,
sum(
l_extendedprice * (1 - l_discount)
) AS revenue,
o_shippriority
FROM
orders
LEFT JOIN lineitem ON l_orderkey = o_orderkey
WHERE
o_orderdate <= DATE '2024-06-30'
AND o_orderdate >= DATE '2024-05-01'
GROUP BY
l_linestatus,
o_shippriority,
l_suppkey;
```

上記の2つのSQLクエリに基づいて、Aggregateを含むより汎用的なマテリアライズドビューを構築できます。このマテリアライズドビューでは、集約のgroup by次元としてl_partkeyとl_suppkeyの両方を含め、フィルター条件としてo_orderdateを使用します。o_orderdateは、マテリアライズドビューの条件補償で使用されるだけでなく、マテリアライズドビューの集約group by次元にも含める必要があることに注意してください。

このようにマテリアライズドビューを構築した後、Query 1とQuery 2の両方がこのマテリアライズドビューにヒットできます。マテリアライズドビューの定義は以下の通りです:

```sql
CREATE MATERIALIZED VIEW common_agg_mv
BUILD IMMEDIATE REFRESH AUTO ON MANUAL
DISTRIBUTED BY RANDOM BUCKETS 2
AS
SELECT
l_linestatus,
sum(
l_extendedprice * (1 - l_discount)
) AS revenue,
o_shippriority,
l_suppkey,
l_partkey,
o_orderdate
FROM
orders
LEFT JOIN lineitem ON l_orderkey = o_orderkey
GROUP BY
l_linestatus,
o_shippriority,
l_suppkey,
l_partkey,
o_orderdate;
```

3. Filterについて

同じフィールドに対するフィルターがクエリに頻繁に現れる場合、マテリアライズドビューに対応するFilterを追加することで、マテリアライズドビュー内のデータ量を削減し、クエリがマテリアライズドビューにヒットした際のパフォーマンスを向上させることができます。

マテリアライズドビューのFilterは、クエリに現れるFilterよりも少なくする必要があり、クエリのFilterはマテリアライズドビューのFilterを含む必要があることに注意してください。例えば、クエリが a > 10 and b > 5 の場合、マテリアライズドビューはFilterを持たないか、もしくはFilterを持つ場合は、クエリよりも大きなデータ範囲でaとbをフィルタリングする必要があります。例えば a > 5 and b > 5b > 0、または単に a > 5 などです。

4. 計算式について

case whenや文字列処理関数のような例を挙げると、これらの式計算は非常にパフォーマンス集約的です。これらをマテリアライズドビューで事前計算できる場合、透明な書き換えを通じて事前計算されたマテリアライズドビューを使用することで、クエリパフォーマンスを向上させることができます。

マテリアライズドビューの列数はあまり多くしないことを推奨します。クエリが複数のフィールドを使用する場合、単一のマテリアライズドビューに過度に多くの列を含めることを避け、初期のSQLパターングループ化に基づいて異なる列に対応するマテリアライズドビューを構築する必要があります。

使用シナリオ

シナリオ1:クエリ高速化

BIレポートシナリオやその他の高速化シナリオにおいて、ユーザーはクエリの応答時間に敏感で、通常は数秒で結果が返されることを要求します。クエリは通常、複数のTable結合に続く集約計算を含み、これらは大量のコンピューティングリソースを消費し、時には適時性を保証することが困難になります。非同期マテリアライズドビューはこれを適切に処理でき、直接クエリと透明な書き換えの両方をサポートし、オプティマイザーが書き換えアルゴリズムとコストモデルに基づいて最適なマテリアライズドビューを自動的に選択してリクエストに応答します。

ユースケース1:マルチTable結合集約クエリ高速化

より汎用的なマテリアライズドビューを構築することで、マルチTable結合集約クエリを高速化できます。

以下の3つのクエリSQLを例に取ります:

クエリ1:

SELECT 
l_linestatus,
l_extendedprice * (1 - l_discount)
o_shippriority
FROM
orders
LEFT JOIN lineitem ON l_orderkey = o_orderkey
WHERE
o_orderdate <= DATE '2024-06-30'
AND o_orderdate >= DATE '2024-05-01';

Query 2:

SELECT 
l_linestatus,
sum(
l_extendedprice * (1 - l_discount)
) AS revenue,
o_orderdate,
o_shippriority
FROM
orders
LEFT JOIN lineitem ON l_orderkey = o_orderkey
WHERE
o_orderdate <= DATE '2024-06-30'
AND o_orderdate >= DATE '2024-05-01'
GROUP BY
l_linestatus,
o_orderdate,
o_shippriority;

Query 3:

SELECT 
l_linestatus,
l_extendedprice * (1 - l_discount),
o_orderdate,
o_shippriority
FROM
orders
LEFT JOIN lineitem ON l_orderkey = o_orderkey;

上記のクエリに対して、すべてのクエリを満たすために以下のmaterialized viewを構築できます。

materialized viewの定義では、Query 1とQuery 2からフィルター条件を削除してより汎用的なJoinを取得し、l_extendedprice * (1 - l_discount)の式を事前計算します。これにより、クエリがmaterialized viewにヒットした際に式の計算を省略できます:

CREATE MATERIALIZED VIEW common_join_mv
BUILD IMMEDIATE REFRESH AUTO ON MANUAL
DISTRIBUTED BY RANDOM BUCKETS 2
AS
SELECT
l_linestatus,
l_extendedprice * (1 - l_discount),
o_orderdate,
o_shippriority
FROM
orders
LEFT JOIN lineitem ON l_orderkey = o_orderkey;

上記のマテリアライズドビューがQuery 2の高速化パフォーマンス要件を満たせない場合、集約マテリアライズドビューを構築することができます。汎用性を保つため、o_orderdateフィールドのフィルター条件を削除することができます:

CREATE MATERIALIZED VIEW target_agg_mv
BUILD IMMEDIATE REFRESH AUTO ON MANUAL
DISTRIBUTED BY RANDOM BUCKETS 2
AS
SELECT
l_linestatus,
sum(
l_extendedprice * (1 - l_discount)
) AS revenue,
o_orderdate,
o_shippriority
FROM
orders
LEFT JOIN lineitem ON l_orderkey = o_orderkey
GROUP BY
l_linestatus,
o_orderdate,
o_shippriority;

Use Case 2: ログ クエリ加速

ログクエリ高速化のシナリオでは、非同期マテリアライズドビューのみの使用に限定せず、同期マテリアライズドビューと組み合わせることが推奨されます。

一般的に、ベースTableはパーティション化されたTableで、多くの場合時間単位でパーティション化されており、単一Tableの集約クエリで、フィルター条件は通常時間といくつかのフラグビットに基づきます。クエリの応答速度が要件を満たせない場合、通常は同期マテリアライズドビューを構築して高速化することができます。

例えば、ベースTableの定義は以下のようになる可能性があります:

CREATE TABLE IF NOT EXISTS test (
`app_name` VARCHAR(64) NULL COMMENT 'identifier',
`event_id` VARCHAR(128) NULL COMMENT 'identifier',
`decision` VARCHAR(32) NULL COMMENT 'enum value',
`time` DATETIME NULL COMMENT 'query time',
`id` VARCHAR(35) NOT NULL COMMENT 'od',
`code` VARCHAR(64) NULL COMMENT 'identifier',
`event_type` VARCHAR(32) NULL COMMENT 'event type'
)
DUPLICATE KEY(app_name, event_id)
PARTITION BY RANGE(time)
(
FROM ("2024-07-01 00:00:00") TO ("2024-07-15 00:00:00") INTERVAL 1 HOUR
)
DISTRIBUTED BY HASH(event_id)
BUCKETS 3;

マテリアライズドビューは分単位でデータを集約でき、これにより一定の集約効果を達成することも可能です。例:

CREATE MATERIALIZED VIEW sync_mv
AS
SELECT
decision,
code,
app_name,
event_id,
event_type,
date_trunc(time, 'minute'),
DATE_FORMAT(
`time`, '%Y-%m-%d'
),
cast(FLOOR(MINUTE(time) / 15) as decimal(9, 0)),
count(id) as cnt
from
test
group by
code,
app_name,
event_id,
event_type,
date_trunc(time, 'minute'),
decision,
DATE_FORMAT(time, '%Y-%m-%d'),
cast(FLOOR(MINUTE(`time`) / 15) as decimal(9, 0));

クエリ文は以下のようになる場合があります:

SELECT 
decision,
CONCAT(
CONCAT(
DATE_FORMAT(
`time`, '%Y-%m-%d'
),
'',
LPAD(
cast(FLOOR(MINUTE(`time`) / 15) as decimal(9, 0)) * 15,
5,
'00'
),
':00'
)
) as time,
count(id) as cnt
from
test
where
date_trunc(time, 'minute') BETWEEN '2024-07-02 18:00:00'
AND '2024-07-03 20:00:00'
group by
decision,
DATE_FORMAT(
`time`, "%Y-%m-%d"
),
cast(FLOOR(MINUTE(`time`) / 15) as decimal(9, 0));

シナリオ2: データモデリング(ETL)

データ分析作業では、複数のTableの結合と集約が頻繁に必要となりますが、これは通常、複雑で頻繁に繰り返されるクエリを伴うプロセスです。この種のクエリは、高いクエリレイテンシや高いリソース消費の問題を引き起こす可能性があります。しかし、非同期マテリアライズドビューを使用して階層化されたデータモデルを構築することで、これらの問題を適切に回避できます。既存のマテリアライズドビューに基づいて、より高レベルのマテリアライズドビューを作成でき(バージョン2.1.3以降でサポート)、さまざまな要件に柔軟に対応できます。

異なるレベルのマテリアライズドビューには、それぞれ独自のトリガー方法を設定できます。例えば:

  • 第1層のマテリアライズドビューを定期的にリフレッシュするように設定し、第2層をトリガーリフレッシュに設定できます。この方法では、第1層のマテリアライズドビューがリフレッシュを完了すると、自動的に第2層のマテリアライズドビューのリフレッシュがトリガーされます。
  • 各層のマテリアライズドビューが定期的にリフレッシュするように設定されている場合、第2層のマテリアライズドビューがリフレッシュする際、第1層のマテリアライズドビューのデータがベースTableと同期されているかどうかは考慮されず、単に第1層のマテリアライズドビューのデータを処理して第2層に同期します。

次に、TPC-Hデータセットを使用してデータモデリングにおける非同期マテリアライズドビューの応用を説明します。地域と国別の月次注文数量と利益の分析を例として取り上げます:

元のクエリ(マテリアライズドビューを使用しない場合):

SELECT
n_name,
date_trunc(o.o_orderdate, 'month') as month,
count(distinct o.o_orderkey) as order_count,
sum(l.l_extendedprice * (1 - l.l_discount)) as revenue
FROM orders o
JOIN lineitem l ON o.o_orderkey = l.l_orderkey
JOIN customer c ON o.o_custkey = c.c_custkey
JOIN nation n ON c.c_nationkey = n.n_nationkey
JOIN region r ON n.n_regionkey = r.r_regionkey
GROUP BY n_name, month;

階層化モデリングにおける非同期マテリアライズドビューの使用:

DWD層(詳細データ)の構築、注文詳細ワイドTableの処理

CREATE MATERIALIZED VIEW dwd_order_detail
BUILD IMMEDIATE REFRESH AUTO ON COMMIT
DISTRIBUTED BY RANDOM BUCKETS 16
AS
select
o.o_orderkey,
o.o_custkey,
o.o_orderstatus,
o.o_totalprice,
o.o_orderdate,
c.c_name,
c.c_nationkey,
n.n_name as nation_name,
r.r_name as region_name,
l.l_partkey,
l.l_quantity,
l.l_extendedprice,
l.l_discount,
l.l_tax
from orders o
join customer c on o.o_custkey = c.c_custkey
join nation n on c.c_nationkey = n.n_nationkey
join region r on n.n_regionkey = r.r_regionkey
join lineitem l on o.o_orderkey = l.l_orderkey;

DWSレイヤー(サマリデータ)を構築し、日次注文サマリを実行する

CREATE MATERIALIZED VIEW dws_daily_sales
BUILD IMMEDIATE REFRESH AUTO ON COMMIT
DISTRIBUTED BY RANDOM BUCKETS 16
AS
select
date_trunc(o_orderdate, 'month') as month,
nation_name,
region_name,
bitmap_union(to_bitmap(o_orderkey)) as order_count,
sum(l_extendedprice * (1 - l_discount)) as net_revenue
from dwd_order_detail
group by
date_trunc(o_orderdate, 'month'),
nation_name,
region_name;

最適化されたクエリは、マテリアライズドビューを使用して以下の通りです:

SELECT
nation_name,
month,
bitmap_union_count(order_count),
sum(net_revenue) as revenue
FROM dws_daily_sales
GROUP BY nation_name, month;

シナリオ3: Lake-Warehouse統合フェデレーテッドデータクエリ

現代のデータアーキテクチャにおいて、企業はデータストレージコストとクエリパフォーマンスのバランスを取るため、しばしばlake-warehouse統合設計を採用しています。このアーキテクチャの下では、2つの主要な課題が頻繁に発生します:

  • 限定的なクエリパフォーマンス:データレイクからのデータを頻繁にクエリする際、ネットワークレイテンシやサードパーティサービスによってパフォーマンスが影響を受ける可能性があり、クエリの遅延を引き起こし、ユーザーエクスペリエンスに影響を与えます。
  • データレイヤーモデリングの複雑性:データレイクからリアルタイムデータウェアハウスへのデータフローと変換プロセスにおいて、通常複雑なETLプロセスが必要となり、メンテナンスコストと開発難易度が増加します。

Doris非同期マテリアライズドビューを使用することで、これらの課題を効果的に解決できます:

  • 透過的書き換えによるクエリ高速化:よく使用されるデータレイククエリの結果をDoris内部ストレージにマテリアライズし、透過的書き換えを使用してクエリパフォーマンスを効果的に向上させます。
  • レイヤーモデリングの簡素化:データレイク内のTableに基づいてマテリアライズドビューの作成をサポートし、データレイクからリアルタイムデータウェアハウスへの便利な変換を可能にし、データモデリングプロセスを大幅に簡素化します。

例えば、Hiveを使用する場合:

TPC-Hデータセットを使用したHiveベースのCatalogの作成

CREATE CATALOG hive_catalog PROPERTIES (
'type'='hms', -- hive meta store address
'hive.metastore.uris' = 'thrift://172.21.0.1:7004'
);

Hive Catalogに基づいてマテリアライズドビューを作成する

-- Materialized views can only be created on 内部カタログ, switch to 内部カタログ
switch internal;
create database hive_mv_db;
use hive_mv_db;

CREATE MATERIALIZED VIEW external_hive_mv
BUILD IMMEDIATE REFRESH AUTO ON MANUAL
DISTRIBUTED BY RANDOM BUCKETS 12
AS
SELECT
n_name,
o_orderdate,
sum(l_extendedprice * (1 - l_discount)) AS revenue
FROM
customer,
orders,
lineitem,
supplier,
nation,
region
WHERE
c_custkey = o_custkey
AND l_orderkey = o_orderkey
AND l_suppkey = s_suppkey
AND c_nationkey = s_nationkey
AND s_nationkey = n_nationkey
AND n_regionkey = r_regionkey
AND r_name = 'ASIA'
GROUP BY
n_name,
o_orderdate;

以下のクエリを実行してください。このクエリは、透過的な書き換えによってマテリアライズドビューを自動的に使用して高速化されます。

SELECT
n_name,
sum(l_extendedprice * (1 - l_discount)) AS revenue
FROM
customer,
orders,
lineitem,
supplier,
nation,
region
WHERE
c_custkey = o_custkey
AND l_orderkey = o_orderkey
AND l_suppkey = s_suppkey
AND c_nationkey = s_nationkey
AND s_nationkey = n_nationkey
AND n_regionkey = r_regionkey
AND r_name = 'ASIA'
AND o_orderdate >= DATE '1994-01-01'
AND o_orderdate < DATE '1994-01-01' + INTERVAL '1' YEAR
GROUP BY
n_name
ORDER BY
revenue DESC;
Note

Dorisは現在、Hive以外の外部Tableのデータ変更を検出できません。外部Tableデータに不整合がある場合、マテリアライズドビューを使用するとデータの不整合が発生する可能性があります。次のスイッチは、透過的リライトに参加するマテリアライズドビューに外部Tableを含めることを許可するかどうかを示します(デフォルトはfalse)。データの不整合を受け入れるか、定期的なリフレッシュによって外部Tableデータの整合性を確保できる場合は、このスイッチをtrueに設定できます。 外部Tableを含むマテリアライズドビューを透過的リライトに使用できるかどうかを設定します。デフォルトでは許可されていませんが、データの不整合を受け入れるか、データの整合性を自分で保証できる場合は有効にできます。

SET materialized_view_rewrite_enable_contain_external_table = true;

マテリアライズドビューがMaterializedViewRewriteSuccessButNotChoseステータスの場合、リライトは成功したがCBOによってプランが選択されなかったことを意味します。これは外部Tableの統計情報が不完全なことが原因の可能性があります。 統計情報のためにファイルリストから行数を取得することを有効にします

SET enable_get_row_count_from_file_list = true;

外部Tableの統計情報を表示して、それらが完全かどうかを確認します

SHOW TABLE STATS external_table_name;

シナリオ4:書き込み効率の向上、リソース競合の削減

高スループットなデータ書き込みシナリオでは、システムの安定性と効率的なデータ処理が同様に重要です。非同期マテリアライズドビューの柔軟なリフレッシュ戦略により、ユーザーは特定のシナリオに基づいて適切なリフレッシュ方法を選択でき、書き込み圧力を軽減し、リソース競合を回避できます。

同期マテリアライズドビューと比較して、非同期マテリアライズドビューは3つの柔軟なリフレッシュ戦略を提供します:手動トリガー、トリガーベース、定期トリガー。ユーザーはシナリオ要件に基づいて適切なリフレッシュ戦略を選択できます。ベースTableデータが変更されても、即座にマテリアライズドビューのリフレッシュがトリガーされることはなく、遅延リフレッシュがリソース圧力を軽減し、書き込みリソース競合を効果的に回避します。

以下に示すように、選択されたリフレッシュ方法は定期リフレッシュで、2時間ごとにリフレッシュされます。ordersとlineitemがデータをインポートしても、即座にマテリアライズドビューのリフレッシュはトリガーされません。

CREATE MATERIALIZED VIEW common_schedule_join_mv
BUILD IMMEDIATE REFRESH AUTO ON SCHEDULE EVERY 2 HOUR
DISTRIBUTED BY RANDOM BUCKETS 16
AS
SELECT
l_linestatus,
l_extendedprice * (1 - l_discount),
o_orderdate,
o_shippriority
FROM
orders
LEFT JOIN lineitem ON l_orderkey = o_orderkey;

透明書き換えはクエリSQLを書き換えてクエリの高速化を実現でき、同時にimport SQLを書き換えてインポート効率を向上させることもできます。バージョン2.1.6以降、マテリアライズドビューとベースTableのデータが強い一貫性を持つ場合、Insert IntoやInsert OverwriteなどのDML操作を透明に書き換えることができ、データインポートシナリオのパフォーマンスが大幅に向上します。

  1. Insert Intoデータ用のターゲットTableを作成する
CREATE TABLE IF NOT EXISTS target_table  (
orderdate DATE NOT NULL,
shippriority INTEGER NOT NULL,
linestatus CHAR(1) NOT NULL,
sale DECIMALV3(15,2) NOT NULL
)
DUPLICATE KEY(orderdate, shippriority)
DISTRIBUTED BY HASH(shippriority) BUCKETS 3;
  1. common_schedule_join_mv
CREATE MATERIALIZED VIEW common_schedule_join_mv
BUILD IMMEDIATE REFRESH AUTO ON SCHEDULE EVERY 2 HOUR
DISTRIBUTED BY RANDOM BUCKETS 16
AS
SELECT
l_linestatus,
l_extendedprice * (1 - l_discount),
o_orderdate,
o_shippriority
FROM
orders
LEFT JOIN lineitem ON l_orderkey = o_orderkey;

書き換え前のimport文:

INSERT INTO target_table
SELECT
o_orderdate,
o_shippriority,
l_linestatus,
l_extendedprice * (1 - l_discount)
FROM
orders
LEFT JOIN lineitem ON l_orderkey = o_orderkey;

透明な書き換え後、ステートメントは次のようになります:

INSERT INTO target_table
SELECT *
FROM common_schedule_join_mv;

注意点:DML操作がデータ変更を検出できない外部Tableを含む場合、透明な書き換えにより最新のベースTableデータがリアルタイムでターゲットTableにインポートされない可能性があります。ユーザーがデータの不整合を受け入れることができる、または自身でデータの整合性を保証できる場合は、以下のスイッチを有効にできます:

DMLにおいて、materialized viewがリアルタイムでデータを検出できない外部Tableを含む場合に、構造情報に基づいたmaterialized view透明書き換えを有効にするかどうか、デフォルトは無効

SET enable_dml_materialized_view_rewrite_when_base_table_unawareness = true;