Materialized view

    The emergence of materialized views is mainly to satisfy users. It can analyze any dimension of the original detailed data, but also can quickly analyze and query fixed dimensions.

    • Analyze requirements to cover both detailed data query and fixed-dimensional query.
    • The query only involves a small part of the columns or rows in the table.
    • The query contains some time-consuming processing operations, such as long-time aggregation operations.
    • The query needs to match different prefix indexes.

    Advantage

    • For those queries that frequently use the same sub-query results repeatedly, the performance is greatly improved
    • Doris automatically maintains the data of the materialized view, whether it is a new import or delete operation, it can ensure the data consistency of the base table and the materialized view table. No need for any additional labor maintenance costs.
    • When querying, it will automatically match the optimal materialized view and read data directly from the materialized view.

    Automatic maintenance of materialized view data will cause some maintenance overhead, which will be explained in the limitations of materialized views later.

    Materialized View VS Rollup

    Before the materialized view function, users generally used the Rollup function to improve query efficiency through pre-aggregation. However, Rollup has certain limitations. It cannot do pre-aggregation based on the detailed model.

    Materialized views cover the functions of Rollup while also supporting richer aggregate functions. So the materialized view is actually a superset of Rollup.

    In other words, the functions previously supported by the syntax can now be implemented by CREATE MATERIALIZED VIEW.

    The Doris system provides a complete set of DDL syntax for materialized views, including creating, viewing, and deleting. The syntax of DDL is consistent with PostgreSQL and Oracle.

    Here you must first decide what kind of materialized view to create based on the characteristics of your query statement. This is not to say that your materialized view definition is exactly the same as one of your query statements. There are two principles here:

    1. Abstract from the query statement, the grouping and aggregation methods shared by multiple queries are used as the definition of the materialized view.
    2. It is not necessary to create materialized views for all dimension combinations.

    First of all, the first point, if a materialized view is abstracted, and multiple queries can be matched to this materialized view. This materialized view works best. Because the maintenance of the materialized view itself also consumes resources.

    If the materialized view only fits a particular query, and other queries do not use this materialized view. As a result, the materialized view is not cost-effective, which not only occupies the storage resources of the cluster, but cannot serve more queries.

    Therefore, users need to combine their own query statements and data dimension information to abstract the definition of some materialized views.

    The second point is that in the actual analysis query, not all dimensional analysis will be covered. Therefore, it is enough to create a materialized view for the commonly used combination of dimensions, so as to achieve a space and time balance.

    Creating a materialized view is an asynchronous operation, which means that after the user successfully submits the creation task, Doris will calculate the existing data in the background until the creation is successful.

    The specific syntax can be viewed through the following command:

    Support aggregate functions

    The aggregate functions currently supported by the materialized view function are:

    • SUM, MIN, MAX (Version 0.12)

    • COUNT, BITMAP_UNION, HLL_UNION (Version 0.13)

    • The form of BITMAP_UNION must be: BITMAP_UNION(TO_BITMAP(COLUMN)) The column type can only be an integer (largeint also does not support), or BITMAP_UNION(COLUMN) and the base table is an AGG model.

    • The form of HLL_UNION must be: HLL_UNION(HLL_HASH(COLUMN)) The column type cannot be DECIMAL, or HLL_UNION(COLUMN) and the base table is an AGG model.

    In order to ensure the data consistency between the materialized view table and the Base table, Doris will import, delete and other operations on the Base table are synchronized to the materialized view table. And through incremental update to improve update efficiency. To ensure atomicity through transaction.

    For example, if the user inserts data into the base table through the INSERT command, this data will be inserted into the materialized view synchronously. When both the base table and the materialized view table are written successfully, the INSERT command will return successfully. To

    Query automatic matching

    After the materialized view is successfully created, the user’s query does not need to be changed, that is, it is still the base table of the query. Doris will automatically select an optimal materialized view based on the current query statement, read data from the materialized view and calculate it.

    Users can use the EXPLAIN command to check whether the current query uses a materialized view.

    After the aggregation functions of bitmap and hll match the materialized view in the query, the aggregation operator of the query will be rewritten according to the table structure of the materialized view. See example 2 for details.

    Check what materialized views the current table has, and what their table structure is. Through the following command:

    1. MySQL [test]> desc mv_test all;
    2. +-----------+---------------+-----------------+----------+------+-------+---------+--------------+
    3. | IndexName | IndexKeysType | Field | Type | Null | Key | Default | Extra |
    4. +-----------+---------------+-----------------+----------+------+-------+---------+--------------+
    5. | mv_test | DUP_KEYS | k1 | INT | Yes | true | NULL | |
    6. | | | k2 | BIGINT | Yes | true | NULL | |
    7. | | | k3 | LARGEINT | Yes | true | NULL | |
    8. | | | k4 | SMALLINT | Yes | false | NULL | NONE |
    9. | | | | | | | | |
    10. | mv_2 | AGG_KEYS | k2 | BIGINT | Yes | true | NULL | |
    11. | | | k4 | SMALLINT | Yes | false | NULL | MIN |
    12. | | | k1 | INT | Yes | false | NULL | MAX |
    13. | | | | | | | | |
    14. | mv_3 | AGG_KEYS | k1 | INT | Yes | true | NULL | |
    15. | | | to_bitmap(`k2`) | BITMAP | No | false | | BITMAP_UNION |
    16. | | | | | | | | |
    17. | mv_1 | AGG_KEYS | k4 | SMALLINT | Yes | true | NULL | |
    18. | | | k1 | BIGINT | Yes | false | NULL | SUM |
    19. | | | k3 | LARGEINT | Yes | false | NULL | SUM |
    20. | | | k2 | BIGINT | Yes | false | NULL | MIN |
    21. +-----------+---------------+-----------------+----------+------+-------+---------+--------------+

    You can see that the current mv_test table has three materialized views: mv_1, mv_2 and mv_3, and their table structure.

    Delete materialized view

    If the user no longer needs the materialized view, you can delete the materialized view by ‘DROP’ commen.

    The specific syntax can be viewed through the following command:

    1. HELP DROP MATERIALIZED VIEW

    Best Practice 1

    The use of materialized views is generally divided into the following steps:

    1. Create a materialized view
    2. Asynchronously check whether the materialized view has been constructed
    3. Query and automatically match materialized views

    First is the first step: Create a materialized view

    Assume that the user has a sales record list, which stores the transaction id, salesperson, sales store, sales time, and amount of each transaction. The table building statement is:

    1. create table sales_records(record_id int, seller_id int, store_id int, sale_date date, sale_amt bigint) distributed by hash(record_id) properties("replication_num" = "1");

    The table structure of this sales_records is as follows:

    1. MySQL [test]> desc sales_records;
    2. +-----------+--------+------+-------+---------+--- ----+
    3. | Field | Type | Null | Key | Default | Extra |
    4. +-----------+--------+------+-------+---------+--- ----+
    5. | record_id | INT | Yes | true | NULL | |
    6. | seller_id | INT | Yes | true | NULL | |
    7. | store_id | INT | Yes | true | NULL | |
    8. | sale_date | DATE | Yes | false | NULL | NONE |
    9. | sale_amt | BIGINT | Yes | false | NULL | NONE |
    10. +-----------+--------+------+-------+---------+--- ----+

    At this time, if the user often performs an analysis query on the sales volume of different stores, you can create a materialized view for the sales_records table to group the sales stores and sum the sales of the same sales stores. The creation statement is as follows:

    1. MySQL [test]> create materialized view store_amt as select store_id, sum(sale_amt) from sales_records group by store_id;

    The backend returns to the following figure, indicating that the task of creating a materialized view is submitted successfully.

    1. Query OK, 0 rows affected (0.012 sec)

    Step 2: Check whether the materialized view has been built

    Since the creation of a materialized view is an asynchronous operation, after the user submits the task of creating a materialized view, he needs to asynchronously check whether the materialized view has been constructed through a command. The command is as follows:

    In this command, db_name is a parameter, you need to replace it with your real db name. The result of the command is to display all the tasks of creating a materialized view of this db. The results are as follows:

    1. +-------+---------------+---------------------+--- ------------------+---------------+--------------- --+----------+---------------+-----------+-------- -------------------------------------------------- -------------------------------------------------- -------------+----------+---------+
    2. | JobId | TableName | CreateTime | FinishedTime | BaseIndexName | RollupIndexName | RollupId | TransactionId | State | Msg | Progress | Timeout |
    3. +-------+---------------+---------------------+--- ------------------+---------------+--------------- --+----------+---------------+-----------+-------- -------------------------------------------------- -------------------------------------------------- -------------+----------+---------+
    4. | 22036 | sales_records | 2020-07-30 20:04:28 | 2020-07-30 20:04:57 | sales_records | store_amt | 22037 | 5008 | FINISHED | | NULL | 86400 |
    5. +-------+---------------+---------------------+--- ------------------+---------------+--------------- --+----------+---------------+-----------+-------- ----------------------------------------

    Among them, TableName refers to which table the data of the materialized view comes from, and RollupIndexName refers to the name of the materialized view. One of the more important indicators is State.

    When the State of the task of creating a materialized view has become FINISHED, it means that the materialized view has been created successfully. This means that it is possible to automatically match this materialized view when querying.

    Step 3: Query

    After the materialized view is created, when users query the sales volume of different stores, they will directly read the aggregated data from the materialized view store_amt just created. To achieve the effect of improving query efficiency.

    The user’s query still specifies the query sales_records table, for example:

    1. SELECT store_id, sum(sale_amt) FROM sales_records GROUP BY store_id;

    The above query will automatically match store_amt. The user can use the following command to check whether the current query matches the appropriate materialized view.

    1. EXPLAIN SELECT store_id, sum(sale_amt) FROM sales_records GROUP BY store_id;
    2. +-----------------------------------------------------------------------------+
    3. +-----------------------------------------------------------------------------+
    4. | PLAN FRAGMENT 0 |
    5. | OUTPUT EXPRS:<slot 2> `store_id` | <slot 3> sum(`sale_amt`) |
    6. | PARTITION: UNPARTITIONED |
    7. | |
    8. | RESULT SINK |
    9. | |
    10. | 4:EXCHANGE |
    11. | |
    12. | PLAN FRAGMENT 1 |
    13. | OUTPUT EXPRS: |
    14. | PARTITION: HASH_PARTITIONED: <slot 2> `store_id` |
    15. | |
    16. | STREAM DATA SINK |
    17. | EXCHANGE ID: 04 |
    18. | |
    19. | 3:AGGREGATE (merge finalize) |
    20. | | output: sum(<slot 3> sum(`sale_amt`)) |
    21. | | group by: <slot 2> `store_id` |
    22. | | |
    23. | 2:EXCHANGE |
    24. | |
    25. | PLAN FRAGMENT 2 |
    26. | OUTPUT EXPRS: |
    27. | PARTITION: RANDOM |
    28. | |
    29. | STREAM DATA SINK |
    30. | EXCHANGE ID: 02 |
    31. | HASH_PARTITIONED: <slot 2> `store_id` |
    32. | |
    33. | 1:AGGREGATE (update serialize) |
    34. | | STREAMING |
    35. | | output: sum(`sale_amt`) |
    36. | | group by: `store_id` |
    37. | | |
    38. | 0:OlapScanNode |
    39. | TABLE: sales_records |
    40. | PREAGGREGATION: ON |
    41. | partitions=1/1 |
    42. | rollup: store_amt |
    43. | tabletRatio=10/10 |
    44. | tabletList=22038,22040,22042,22044,22046,22048,22050,22052,22054,22056 |
    45. | cardinality=0 |
    46. | avgRowSize=0.0 |
    47. | numNodes=1 |
    48. +-----------------------------------------------------------------------------+
    49. 45 rows in set (0.006 sec)

    The final thing is the rollup attribute in OlapScanNode. You can see that the rollup of the current query shows store_amt. That is to say, the query has been correctly matched to the materialized view store_amt, and data is read directly from the materialized view.

    Best Practice 2 PV,UV

    Business scenario: Calculate the UV and PV of advertising

    Use the following statement to first create a table that stores the details of the advertisement click data, including the click event of each click, what advertisement was clicked, what channel clicked, and who was the user who clicked.

    1. MySQL [test]> create table advertiser_view_record(time date, advertiser varchar(10), channel varchar(10), user_id int) distributed by hash(time) properties("replication_num" = "1");
    2. Query OK, 0 rows affected (0.014 sec)

    The original ad click data table structure is:

    1. MySQL [test]> desc advertiser_view_record;
    2. +------------+-------------+------+-------+---------+-------+
    3. | Field | Type | Null | Key | Default | Extra |
    4. +------------+-------------+------+-------+---------+-------+
    5. | time | DATE | Yes | true | NULL | |
    6. | advertiser | VARCHAR(10) | Yes | true | NULL | |
    7. | channel | VARCHAR(10) | Yes | false | NULL | NONE |
    8. | user_id | INT | Yes | false | NULL | NONE |
    9. +------------+-------------+------+-------+---------+-------+
    10. 4 rows in set (0.001 sec)
    1. Create a materialized view

      Since the user wants to query the UV value of the advertisement, that is, a precise de-duplication of users of the same advertisement is required, the user’s query is generally:

      1. SELECT advertiser, channel, count(distinct user_id) FROM advertiser_view_record GROUP BY advertiser, channel;

      For this kind of UV-seeking scene, we can create a materialized view with bitmap_union to achieve a precise deduplication effect in advance.

      In Doris, the result of count(distinct) aggregation is exactly the same as the result of bitmap_union_count aggregation. And bitmap_union_count is equal to the result of bitmap_union to calculate count, so if the query ** involves count(distinct), you can speed up the query by creating a materialized view with bitmap_union aggregation.**

      For this case, you can create a materialized view that accurately deduplicate user_id based on advertising and channel grouping.

      *Note: Because the user_id itself is an INT type, it is called bitmap_union directly in Doris. The fields need to be converted to bitmap type through the function to_bitmap first, and then bitmap_union can be aggregated. *

      After the creation is complete, the table structure of the advertisement click schedule and the materialized view table is as follows:

      1. MySQL [test]> desc advertiser_view_record all;
      2. +------------------------+---------------+----------------------+-------------+------+-------+---------+--------------+
      3. | IndexName | IndexKeysType | Field | Type | Null | Key | Default | Extra |
      4. +------------------------+---------------+----------------------+-------------+------+-------+---------+--------------+
      5. | advertiser_view_record | DUP_KEYS | time | DATE | Yes | true | NULL | |
      6. | | | advertiser | VARCHAR(10) | Yes | true | NULL | |
      7. | | | channel | VARCHAR(10) | Yes | false | NULL | NONE |
      8. | | | user_id | INT | Yes | false | NULL | NONE |
      9. | | | | | | | | |
      10. | advertiser_uv | AGG_KEYS | advertiser | VARCHAR(10) | Yes | true | NULL | |
      11. | | | channel | VARCHAR(10) | Yes | true | NULL | |
      12. | | | to_bitmap(`user_id`) | BITMAP | No | false | | BITMAP_UNION |
    2. Automatic query matching

      When the materialized view table is created, when querying the advertisement UV, Doris will automatically query the data from the materialized view advertiser_uv just created. For example, the original query statement is as follows:

      1. SELECT advertiser, channel, count(distinct user_id) FROM advertiser_view_record GROUP BY advertiser, channel;

      After the materialized view is selected, the actual query will be transformed into:

      1. SELECT advertiser, channel, bitmap_union_count(to_bitmap(user_id)) FROM advertiser_uv GROUP BY advertiser, channel;

      Through the EXPLAIN command, you can check whether Doris matches the materialized view:

      1. MySQL [test]> explain SELECT advertiser, channel, count(distinct user_id) FROM advertiser_view_record GROUP BY advertiser, channel;
      2. +-------------------------------------------------------------------------------------------------------------------------------------------------------------------+
      3. | Explain String |
      4. +-------------------------------------------------------------------------------------------------------------------------------------------------------------------+
      5. | PLAN FRAGMENT 0 |
      6. | OUTPUT EXPRS:<slot 7> `advertiser` | <slot 8> `channel` | <slot 9> bitmap_union_count(`default_cluster:test`.`advertiser_view_record`.`mv_bitmap_union_user_id`) |
      7. | PARTITION: UNPARTITIONED |
      8. | |
      9. | RESULT SINK |
      10. | |
      11. | 4:EXCHANGE |
      12. | |
      13. | PLAN FRAGMENT 1 |
      14. | OUTPUT EXPRS: |
      15. | PARTITION: HASH_PARTITIONED: <slot 4> `advertiser`, <slot 5> `channel` |
      16. | |
      17. | STREAM DATA SINK |
      18. | EXCHANGE ID: 04 |
      19. | UNPARTITIONED |
      20. | |
      21. | 3:AGGREGATE (merge finalize) |
      22. | | output: bitmap_union_count(<slot 6> bitmap_union_count(`default_cluster:test`.`advertiser_view_record`.`mv_bitmap_union_user_id`)) |
      23. | | group by: <slot 4> `advertiser`, <slot 5> `channel` |
      24. | | |
      25. | 2:EXCHANGE |
      26. | |
      27. | PLAN FRAGMENT 2 |
      28. | OUTPUT EXPRS: |
      29. | PARTITION: RANDOM |
      30. | |
      31. | STREAM DATA SINK |
      32. | EXCHANGE ID: 02 |
      33. | HASH_PARTITIONED: <slot 4> `advertiser`, <slot 5> `channel` |
      34. | |
      35. | 1:AGGREGATE (update serialize) |
      36. | | STREAMING |
      37. | | output: bitmap_union_count(`default_cluster:test`.`advertiser_view_record`.`mv_bitmap_union_user_id`) |
      38. | | group by: `advertiser`, `channel` |
      39. | | |
      40. | 0:OlapScanNode |
      41. | TABLE: advertiser_view_record |
      42. | PREAGGREGATION: ON |
      43. | partitions=1/1 |
      44. | rollup: advertiser_uv |
      45. | tabletRatio=10/10 |
      46. | tabletList=22084,22086,22088,22090,22092,22094,22096,22098,22100,22102 |
      47. | cardinality=0 |
      48. | avgRowSize=0.0 |
      49. | numNodes=1 |
      50. +-------------------------------------------------------------------------------------------------------------------------------------------------------------------+
      51. 45 rows in set (0.030 sec)

      In the result of EXPLAIN, you can first see that the rollup attribute value of OlapScanNode is advertiser_uv. In other words, the query directly scans the data of the materialized view. The match is successful.

      Secondly, the calculation of count(distinct) for the user_id field is rewritten as bitmap_union_count. That is to achieve the effect of precise deduplication through bitmap.

    Business scenario: matching a richer prefix index

    The user’s original table has three columns (k1, k2, k3). Among them, k1, k2 are prefix index columns. At this time, if the user query condition contains where k1=a and k2=b, the query can be accelerated through the index.

    But in some cases, the user’s filter conditions cannot match the prefix index, such as where k3=c. Then the query speed cannot be improved through the index.

    This problem can be solved by creating a materialized view with k3 as the first column.

    1. Create a materialized view

      1. CREATE MATERIALIZED VIEW mv_1 as SELECT k3, k2, k1 FROM tableA ORDER BY k3;

      After the creation of the above grammar is completed, the complete detail data is retained in the materialized view, and the prefix index of the materialized view is the k3 column. The table structure is as follows:

      1. MySQL [test]> desc tableA all;
      2. +-----------+---------------+-------+------+------+-------+---------+-------+
      3. | IndexName | IndexKeysType | Field | Type | Null | Key | Default | Extra |
      4. +-----------+---------------+-------+------+------+-------+---------+-------+
      5. | tableA | DUP_KEYS | k1 | INT | Yes | true | NULL | |
      6. | | | k2 | INT | Yes | true | NULL | |
      7. | | | k3 | INT | Yes | true | NULL | |
      8. | | | | | | | | |
      9. | mv_1 | DUP_KEYS | k3 | INT | Yes | true | NULL | |
      10. | | | k2 | INT | Yes | false | NULL | NONE |
      11. | | | k1 | INT | Yes | false | NULL | NONE |
    2. Query matching

      At this time, the query will read data directly from the mv_1 materialized view just created. The materialized view has a prefix index on k3, and query efficiency will also be improved.

    Limitations

    1. The parameter of the aggregate function of the materialized view does not support the expression only supports a single column, for example: sum(a+b) does not support.
    2. If the conditional column of the delete statement does not exist in the materialized view, the delete operation cannot be performed. If you must delete data, you need to delete the materialized view before deleting the data.
    3. Too many materialized views on a single table will affect the efficiency of importing: When importing data, the materialized view and base table data are updated synchronously. If a table has more than 10 materialized view tables, it may cause the import speed to be very high. slow. This is the same as a single import needs to import 10 tables at the same time.
    4. The same column with different aggregate functions cannot appear in a materialized view at the same time. For example, select sum(a), min(a) from table are not supported.
    5. For the Unique Key data model, the materialized view can only change the column order and cannot play the role of aggregation. Therefore, in the Unique Key model, it is not possible to perform coarse-grained aggregation operations on the data by creating a materialized view.

    Error