Generated Columns

This document introduces the concept and usage of generated columns.

Basic concepts

Unlike general columns, the value of the generated column is calculated by the expression in the column definition. When inserting or updating a generated column, you cannot assign a value, but only use DEFAULT.

There are two kinds of generated columns: virtual and stored. A virtual generated column occupies no storage and is computed when it is read. A stored generated column is computed when it is written (inserted or updated) and occupies storage. Compared with the virtual generated columns, the stored generated columns have better read performance, but take up more disk space.

You can create an index on a generated column whether it is virtual or stored.

Usage

One of the main usage of generated columns is to extract data from the JSON data type and indexing the data.

In both MySQL 5.7 and TiDB, columns of type JSON cannot be indexed directly. That is, the following table schema is not supported:

  1. CREATE TABLE person (
  2. id INT NOT NULL AUTO_INCREMENT PRIMARY KEY,
  3. name VARCHAR(255) NOT NULL,
  4. address_info JSON,
  5. KEY (address_info)
  6. );

To index a JSON column, you must extract it as a generated column first.

Using the city field in address_info as an example, you can create a virtual generated column and add an index for it:

  1. CREATE TABLE person (
  2. id INT NOT NULL AUTO_INCREMENT PRIMARY KEY,
  3. name VARCHAR(255) NOT NULL,
  4. address_info JSON,
  5. city VARCHAR(64) AS (JSON_UNQUOTE(JSON_EXTRACT(address_info, '$.city'))), -- virtual generated column
  6. -- city VARCHAR(64) AS (JSON_UNQUOTE(JSON_EXTRACT(address_info, '$.city'))) VIRTUAL, -- virtual generated column
  7. -- city VARCHAR(64) AS (JSON_UNQUOTE(JSON_EXTRACT(address_info, '$.city'))) STORED, -- stored generated column
  8. KEY (city)
  9. );

In this table, the city column is a virtual generated column and has an index. The following query can use the index to speed up the execution:

  1. SELECT name, id FROM person WHERE city = 'Beijing';
  1. EXPLAIN SELECT name, id FROM person WHERE city = 'Beijing';
  1. +---------------------------------+---------+-----------+--------------------------------+-------------------------------------------------------------+
  2. | id | estRows | task | access object | operator info |
  3. +---------------------------------+---------+-----------+--------------------------------+-------------------------------------------------------------+
  4. | Projection_4 | 10.00 | root | | test.person.name, test.person.id |
  5. | └─IndexLookUp_10 | 10.00 | root | | |
  6. | ├─IndexRangeScan_8(Build) | 10.00 | cop[tikv] | table:person, index:city(city) | range:["Beijing","Beijing"], keep order:false, stats:pseudo |
  7. | └─TableRowIDScan_9(Probe) | 10.00 | cop[tikv] | table:person | keep order:false, stats:pseudo |
  8. +---------------------------------+---------+-----------+--------------------------------+-------------------------------------------------------------+

From the query execution plan, it can be seen that the city index is used to read the HANDLE of the row that meets the condition city ='Beijing', and then it uses this HANDLE to read the data of the row.

If no data exists at path $.city, JSON_EXTRACT returns NULL. If you want to enforce a constraint that city must be NOT NULL, you can define the virtual generated column as follows:

  1. CREATE TABLE person (
  2. id INT NOT NULL AUTO_INCREMENT PRIMARY KEY,
  3. name VARCHAR(255) NOT NULL,
  4. address_info JSON,
  5. city VARCHAR(64) AS (JSON_UNQUOTE(JSON_EXTRACT(address_info, '$.city'))) NOT NULL,
  6. KEY (city)
  7. );

Validation of generated columns

Both INSERT and UPDATE statements check virtual column definitions. Rows that do not pass validation return errors:

  1. mysql> INSERT INTO person (name, address_info) VALUES ('Morgan', JSON_OBJECT('Country', 'Canada'));
  2. ERROR 1048 (23000): Column 'city' cannot be null

Generated columns index replacement rule

When an expression in a query is strictly equivalent to a generated column with an index, TiDB replaces the expression with the corresponding generated column, so that the optimizer can take that index into account during execution plan construction.

The following example creates a generated column for the expression a+1 and adds an index. The column type of a is int and the column type of a+1 is bigint. If the type of the generated column is set to int, the replacement will not occur. For type conversion rules, see Type Conversion of Expression Evaluation.

  1. create table t(a int);
  2. desc select a+1 from t where a+1=3;
  1. +---------------------------+----------+-----------+---------------+--------------------------------+
  2. | id | estRows | task | access object | operator info |
  3. +---------------------------+----------+-----------+---------------+--------------------------------+
  4. | Projection_4 | 8000.00 | root | | plus(test.t.a, 1)->Column#3 |
  5. | └─TableReader_7 | 8000.00 | root | | data:Selection_6 |
  6. | └─Selection_6 | 8000.00 | cop[tikv] | | eq(plus(test.t.a, 1), 3) |
  7. | └─TableFullScan_5 | 10000.00 | cop[tikv] | table:t | keep order:false, stats:pseudo |
  8. +---------------------------+----------+-----------+---------------+--------------------------------+
  9. 4 rows in set (0.00 sec)
  1. alter table t add column b bigint as (a+1) virtual;
  2. alter table t add index idx_b(b);
  3. desc select a+1 from t where a+1=3;
  1. +------------------------+---------+-----------+-------------------------+---------------------------------------------+
  2. | id | estRows | task | access object | operator info |
  3. +------------------------+---------+-----------+-------------------------+---------------------------------------------+
  4. | IndexReader_6 | 10.00 | root | | index:IndexRangeScan_5 |
  5. | └─IndexRangeScan_5 | 10.00 | cop[tikv] | table:t, index:idx_b(b) | range:[3,3], keep order:false, stats:pseudo |
  6. +------------------------+---------+-----------+-------------------------+---------------------------------------------+
  7. 2 rows in set (0.01 sec)

Generated Columns - 图1

Note

If the expression to be replaced and the generated column are both the string type but with different lengths, you can still replace the expression by setting the system variable tidb_enable_unsafe_substitute to ON. When configuring this system variable, ensure that the value calculated by the generated column strictly satisfies the definition of the generated column. Otherwise, the data might be truncated due to the difference in length, resulting in an incorrect result. See GitHub issue #35490.

Limitations

The current limitations of JSON and generated columns are as follows:

  • You cannot add a stored generated column through ALTER TABLE.
  • You can neither convert a stored generated column to a normal column through the ALTER TABLE statement nor convert a normal column to a stored generated column.
  • You cannot modify the expression of a stored generated column through the ALTER TABLE statement.
  • Not all JSON functions are supported;
  • Currently, the generated column index replacement rule is valid only when the generated column is a virtual generated column. It is not valid on the stored generated column, but the index can still be used by directly using the generated column itself.