TiDB 特有的函数

本文档介绍 TiDB 特有的函数。

TIDB_BOUNDED_STALENESS

TIDB_BOUNDED_STALENESS 是 TiDB 的内部函数,用于指定一个时间范围。用法为 TIDB_BOUNDED_STALENESS(t1, t2),其中 t1 和 t2 为时间范围的两端,支持使用日期时间和时间函数。

使用该函数,TiDB 会在指定的时间范围内选择一个合适的时间戳,该时间戳能保证所访问的副本上不存在开始于这个时间戳之前且还没有提交的相关事务,即能保证所访问的可用副本上执行读取操作而且不会被阻塞。

TIDB_DECODE_KEY

TIDB_DECODE_KEY 函数用于将 TiDB 编码的键输入解码为包含 _tidb_rowidtable_id 的 JSON 结构。你可以在一些系统表和日志输出中找到 TiDB 的编码键。

语法图

TableStmt

TiDB 特有的函数 - 图1

  1. TableStmt ::=
  2. "TIDB_DECODE_KEY(" STR ")"

示例

以下示例中,表 t1 有一个隐藏的 rowid,该 rowid 由 TiDB 生成。语句中使用了 TIDB_DECODE_KEY 函数。结果显示,隐藏的 rowid 被解码后并输出,这是典型的非聚簇主键结果。

  1. SELECT START_KEY, TIDB_DECODE_KEY(START_KEY) FROM information_schema.tikv_region_status WHERE table_name='t1' AND REGION_ID=2\G
  1. *************************** 1. row ***************************
  2. START_KEY: 7480000000000000FF3B5F728000000000FF1DE3F10000000000FA
  3. TIDB_DECODE_KEY(START_KEY): {"_tidb_rowid":1958897,"table_id":"59"}
  4. 1 row in set (0.00 sec)

以下示例中,表 t2 有一个复合聚簇主键。由 JSON 输出可知,输出结果的 handle 项中包含了主键部分两列的信息,即两列的名称和对应的值。

  1. show create table t2\G
  1. *************************** 1. row ***************************
  2. Table: t2
  3. Create Table: CREATE TABLE `t2` (
  4. `id` binary(36) NOT NULL,
  5. `a` tinyint(3) unsigned NOT NULL,
  6. `v` varchar(512) DEFAULT NULL,
  7. PRIMARY KEY (`a`,`id`) /*T![clustered_index] CLUSTERED */
  8. ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_bin
  9. 1 row in set (0.001 sec)
  1. select * from information_schema.tikv_region_status where table_name='t2' limit 1\G
  1. *************************** 1. row ***************************
  2. REGION_ID: 48
  3. START_KEY: 7480000000000000FF3E5F720400000000FF0000000601633430FF3338646232FF2D64FF3531632D3131FF65FF622D386337352DFFFF3830653635303138FFFF61396265000000FF00FB000000000000F9
  4. END_KEY:
  5. TABLE_ID: 62
  6. DB_NAME: test
  7. TABLE_NAME: t2
  8. IS_INDEX: 0
  9. INDEX_ID: NULL
  10. INDEX_NAME: NULL
  11. EPOCH_CONF_VER: 1
  12. EPOCH_VERSION: 38
  13. WRITTEN_BYTES: 0
  14. READ_BYTES: 0
  15. APPROXIMATE_SIZE: 136
  16. APPROXIMATE_KEYS: 479905
  17. REPLICATIONSTATUS_STATE: NULL
  18. REPLICATIONSTATUS_STATEID: NULL
  19. 1 row in set (0.005 sec)
  1. select tidb_decode_key('7480000000000000FF3E5F720400000000FF0000000601633430FF3338646232FF2D64FF3531632D3131FF65FF622D386337352DFFFF3830653635303138FFFF61396265000000FF00FB000000000000F9');
  1. +---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
  2. | tidb_decode_key('7480000000000000FF3E5F720400000000FF0000000601633430FF3338646232FF2D64FF3531632D3131FF65FF622D386337352DFFFF3830653635303138FFFF61396265000000FF00FB000000000000F9') |
  3. +---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
  4. | {"handle":{"a":"6","id":"c4038db2-d51c-11eb-8c75-80e65018a9be"},"table_id":62} |
  5. +---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
  6. 1 row in set (0.001 sec)

MySQL 兼容性

TIDB_DECODE_KEY 是 TiDB 特有的函数,和 MySQL 不兼容。

TIDB_DECODE_PLAN

TIDB_DECODE_PLAN 函数用于解码 TiDB 执行计划。你可以在慢查询日志中找到 TiDB 执行计划。

语法图

TableStmt

TiDB 特有的函数 - 图2

  1. TableStmt ::=
  2. "TIDB_DECODE_PLAN(" STR ")"

示例

  1. SELECT tidb_decode_plan('8QIYMAkzMV83CQEH8E85LjA0CWRhdGE6U2VsZWN0aW9uXzYJOTYwCXRpbWU6NzEzLjHCtXMsIGxvb3BzOjIsIGNvcF90YXNrOiB7bnVtOiAxLCBtYXg6IDU2OC41wgErRHByb2Nfa2V5czogMCwgcnBjXxEpAQwFWBAgNTQ5LglZyGNvcHJfY2FjaGVfaGl0X3JhdGlvOiAwLjAwfQkzLjk5IEtCCU4vQQoxCTFfNgkxXzAJMwm2SGx0KHRlc3QudC5hLCAxMDAwMCkNuQRrdgmiAHsFbBQzMTMuOMIBmQnEDDk2MH0BUgEEGAoyCTQzXzUFVwX1oGFibGU6dCwga2VlcCBvcmRlcjpmYWxzZSwgc3RhdHM6cHNldWRvCTk2ISE2aAAIMTUzXmYA')\G
  1. *************************** 1. row ***************************
  2. tidb_decode_plan('8QIYMAkzMV83CQEH8E85LjA0CWRhdGE6U2VsZWN0aW9uXzYJOTYwCXRpbWU6NzEzLjHCtXMsIGxvb3BzOjIsIGNvcF90YXNrOiB7bnVtOiAxLCBtYXg6IDU2OC41wgErRHByb2Nfa2V5czogMCwgcnBjXxEpAQwFWBAgNTQ5LglZyGNvcHJfY2FjaGVfaGl0X3JhdGlvOiAwLjAwfQkzLjk5IEtCCU4vQQoxCTFfNgkxXz: id task estRows operator info actRows execution info memory disk
  3. TableReader_7 root 319.04 data:Selection_6 960 time:713.1µs, loops:2, cop_task: {num: 1, max: 568.5µs, proc_keys: 0, rpc_num: 1, rpc_time: 549.1µs, copr_cache_hit_ratio: 0.00} 3.99 KB N/A
  4. └─Selection_6 cop[tikv] 319.04 lt(test.t.a, 10000) 960 tikv_task:{time:313.8µs, loops:960} N/A N/A
  5. └─TableFullScan_5 cop[tikv] 960 table:t, keep order:false, stats:pseudo 960 tikv_task:{time:153µs, loops:960} N/A N/A

MySQL 兼容性

TIDB_DECODE_PLAN 是 TiDB 特有的函数,和 MySQL 不兼容。

TIDB_IS_DDL_OWNER

TIDB_IS_DDL_OWNER 函数用于检查你连接的 TiDB 实例是否是 DDL Owner。DDL Owner 代表集群中所有其他节点执行 DDL 语句的 TiDB 实例。

语法图

TableStmt

TiDB 特有的函数 - 图3

  1. TableStmt ::=
  2. "TIDB_IS_DDL_OWNER())"

示例

  1. SELECT tidb_is_ddl_owner();
  1. +---------------------+
  2. | tidb_is_ddl_owner() |
  3. +---------------------+
  4. | 1 |
  5. +---------------------+
  6. 1 row in set (0.00 sec)

MySQL 兼容性

TIDB_IS_DDL_OWNER 是 TiDB 特有的函数,和 MySQL 不兼容。

另请参阅

TIDB_PARSE_TSO

TIDB_PARSE_TSO 函数用于从 TiDB TSO 时间戳中提取物理时间戳。

TSO 指 Time Stamp Oracle,是 PD (Placement Driver) 为每个事务提供的单调递增的时间戳。

TSO 是一串数字,包含以下两部分:

  • 一个物理时间戳
  • 一个逻辑计数器

语法图

TableStmt

TiDB 特有的函数 - 图4

  1. TableStmt ::=
  2. "TIDB_PARSE_TSO(" NUM ")"

示例

  1. BEGIN;
  2. SELECT TIDB_PARSE_TSO(@@tidb_current_ts);
  3. ROLLBACK;
  1. +-----------------------------------+
  2. | TIDB_PARSE_TSO(@@tidb_current_ts) |
  3. +-----------------------------------+
  4. | 2021-05-26 11:33:37.776000 |
  5. +-----------------------------------+
  6. 1 row in set (0.0012 sec)

以上示例使用 TIDB_PARSE_TSO 函数从 tidb_current_ts 会话变量提供的可用时间戳编号中提取物理时间戳。因为每个事务都会分配到时间戳,所以此函数在事务中运行。

MySQL 兼容性

TIDB_PARSE_TSO 是 TiDB 特有的函数,和 MySQL 不兼容。

另请参阅

TIDB_VERSION

TIDB_VERSION 函数用于获取当前连接的 TiDB 服务器版本和构建详细信息。向 GitHub 上提交 issue 时,你可使用此函数获取相关信息。

语法图

TableStmt

TiDB 特有的函数 - 图5

  1. TableStmt ::=
  2. "TIDB_VERSION()"

示例

  1. SELECT TIDB_VERSION()\G
  1. *************************** 1. row ***************************
  2. TIDB_VERSION(): Release Version: v5.1.0-alpha-13-gd5e0ed0aa-dirty
  3. Edition: Community
  4. Git Commit Hash: d5e0ed0aaed72d2f2dfe24e9deec31cb6cb5fdf0
  5. Git Branch: master
  6. UTC Build Time: 2021-05-24 14:39:20
  7. GoVersion: go1.13
  8. Race Enabled: false
  9. TiKV Min Version: v3.0.0-60965b006877ca7234adaced7890d7b029ed1306
  10. Check Table Before Drop: false
  11. 1 row in set (0.00 sec)

MySQL 兼容性

TIDB_VERSION 是 TiDB 特有的函数,和 MySQL 不兼容。如果要求兼容 MySQL,可以使用 VERSION 获取版本信息,但结果不包含详细的构建信息。

TIDB_DECODE_SQL_DIGESTS

TIDB_DECODE_SQL_DIGESTS 函数用于在集群中查询一组 SQL Digest 所对应的 SQL 语句的归一化形式(即去除格式和参数后的形式)。函数接受 1 个或 2 个参数:

  • digests:字符串类型,该参数应符合 JSON 字符串数组的格式,数组中的每个字符串应为一个 SQL Digest。
  • stmtTruncateLength:可选参数,整数类型,用来限制返回结果中每条 SQL 语句的长度,超过指定的长度会被截断。0 表示不限制长度。

返回一个字符串,符合 JSON 字符串数组的格式,数组中的第 i 项为参数 digests 中的第 i 个元素所对应的语句。如果参数 digests 中的某一项不是一个有效的 SQL Digest 或系统无法查询到其对应的 SQL 语句,则返回结果中对应项为 null。如果指定了截断长度(stmtTruncateLength > 0),则返回结果中每条超过该长度的语句,保留前 stmtTruncateLength 个字符,并在尾部增加 "..." 后缀表示发生了截断。如果参数 digestsNULL,则函数的返回值为 NULL

TiDB 特有的函数 - 图6

注意

  • 仅持有 PROCESS 权限的用户可以使用该函数。
  • TIDB_DECODE_SQL_DIGESTS 执行时,TiDB 内部从 Statement Summary 一系列表中查询每个 SQL Digest 所对应的语句,因而并不能保证对任意 SQL Digest 都总是能查询到对应的语句,只有在集群中执行过的语句才有可能被查询到,且是否能查询到受 Statement Summary 表相关配置的影响。有关 Statement Summary 表的详细说明,参见 Statement Summary Tables
  • 该函数开销较大,在行数很多的查询中(比如在规模较大、比较繁忙的集群上查询 information_schema.cluster_tidb_trx 全表时)直接使用该函数可能导致查询运行时间较长。请谨慎使用。
    • 该函数开销大的原因是,其每次被调用时,都会在内部发起对 STATEMENTS_SUMMARYSTATEMENTS_SUMMARY_HISTORYCLUSTER_STATEMENTS_SUMMARYCLUSTER_STATEMENTS_SUMMARY_HISTORY 这几张表的查询,且其中涉及 UNION 操作。且该函数目前不支持向量化,即对于多行数据调用该函数时,对每行都会独立进行一次上述的查询。

语法图

DecodeSQLDigestsExpr

TiDB 特有的函数 - 图7

  1. DecodeSQLDigestsExpr ::=
  2. "TIDB_DECODE_SQL_DIGESTS" "(" digests ( "," stmtTruncateLength )? ")"

示例

  1. set @digests = '["e6f07d43b5c21db0fbb9a31feac2dc599787763393dd5acbfad80e247eb02ad5","38b03afa5debbdf0326a014dbe5012a62c51957f1982b3093e748460f8b00821","e5796985ccafe2f71126ed6c0ac939ffa015a8c0744a24b7aee6d587103fd2f7"]';
  2. select tidb_decode_sql_digests(@digests);
  1. +------------------------------------+
  2. | tidb_decode_sql_digests(@digests) |
  3. +------------------------------------+
  4. | ["begin",null,"select * from `t`"] |
  5. +------------------------------------+
  6. 1 row in set (0.00 sec)

上面的例子中,参数是一个包含 3 个 SQL Digest 的 JSON 数组,其对应的 SQL 语句分别为查询结果中给出的三项。但是其中第二条 SQL Digest 所对应的 SQL 语句未能从集群中找到,因而结果中的第二项为 null

  1. select tidb_decode_sql_digests(@digests, 10);
  1. +---------------------------------------+
  2. | tidb_decode_sql_digests(@digests, 10) |
  3. +---------------------------------------+
  4. | ["begin",null,"select * f..."] |
  5. +---------------------------------------+
  6. 1 row in set (0.01 sec)

上述调用指定了第二个参数(即截断长度)为 10,而查询结果中的第三条语句的长度大于 10,因而仅保留了前 10 个字符,并在尾部添加了 "..." 表示发生了截断。

MySQL 兼容性

TIDB_DECODE_SQL_DIGESTS 是 TiDB 特有的函数,和 MySQL 不兼容。

另请参阅

TIDB_SHARD

TIDB_SHARD 函数用于创建一个 SHARD INDEX 来打散热点索引。SHARD INDEX 是一种以 TIDB_SHARD 函数为前缀的表达式索引。

SHARD INDEX

  • 创建方式:

    使用 uk((tidb_shard(a)), a)) 为字段 a 创建一个 SHARD INDEX。当二级唯一索引 uk((tidb_shard(a)), a)) 的索引字段 a 上存在因单调递增或递减而产生的热点时,索引的前缀 tidb_shard(a) 会打散热点,从而提升集群可扩展性。

  • 适用场景:

    • 二级唯一索引上 key 值存在单调递增或递减导致的写入热点,且该索引包含的列是整型。
    • 业务中 SQL 语句根据该二级索引的全部字段做等值查询,查询可以是单独的 SELECT,也可以是 UPDATEDELETE 等产生的内部查询,等值查询包括 a = 1a IN (1, 2, ......) 两种方式。
  • 使用限制:

    • 非等值查询无法使用索引。
    • 查询条件中 ANDOR 混合且最外层是 AND 算子时无法使用 SHARD INDEX。
    • GROUP BY 无法使用 SHARD INDEX。
    • ORDER BY 无法使用 SHARD INDEX。
    • ON 子句无法使用 SHARD INDEX。
    • WHERE 子查询无法使用 SHARD INDEX。
    • SHARD INDEX 只能打散整型字段的唯一索引。
    • SHARD INDEX 联合索引可能失效。
    • SHARD INDEX 无法走 FastPlan 流程,影响优化器性能。
    • SHARD INDEX 无法使用执行计划缓存。

语法图

TIDBShardExpr

TiDB 特有的函数 - 图8

  1. TIDBShardExpr ::=
  2. "TIDB_SHARD" "(" expr ")"

示例

  • 使用 TIDB_SHARD 函数计算 SHARD 值

    以下示例说明如何使用 TIDB_SHARD 函数计算 12373743746 的 SHARD 值。

    1. SELECT TIDB_SHARD(12373743746);
  • 计算得出 SHARD 值为:

    1. +-------------------------+
    2. | TIDB_SHARD(12373743746) |
    3. +-------------------------+
    4. | 184 |
    5. +-------------------------+
    6. 1 row in set (0.00 sec)
  • 使用 TIDB_SHARD 函数创建 SHARD INDEX

    1. CREATE TABLE test(id INT PRIMARY KEY CLUSTERED, a INT, b INT, UNIQUE KEY uk((tidb_shard(a)), a));

MySQL 兼容性

TIDB_SHARD 是 TiDB 特有的函数,和 MySQL 不兼容。

TIDB_ROW_CHECKSUM

TIDB_ROW_CHECKSUM 函数用于查询行数据的 Checksum 值。该函数只能用于 FastPlan 流程的 SELECT 语句,即你可通过形如 SELECT TIDB_ROW_CHECKSUM() FROM t WHERE id = ?SELECT TIDB_ROW_CHECKSUM() FROM t WHERE id IN (?, ?, ...) 的语句进行查询。

语法图

TableStmt

TiDB 特有的函数 - 图9

  1. TableStmt ::=
  2. "TIDB_ROW_CHECKSUM()"

示例

在 TiDB 中开启行数据 Checksum 功能 tidb_enable_row_level_checksum

  1. SET GLOBAL tidb_enable_row_level_checksum = ON;

创建表 t 并插入数据:

  1. USE test;
  2. CREATE TABLE t (id INT PRIMARY KEY, k INT, c int);
  3. INSERT INTO TABLE t values (1, 10, a);

查询表 tid = 1 的行数据的 Checksum 值:

  1. SELECT *, TIDB_ROW_CHECKSUM() FROM t WHERE id = 1;
  1. +----+------+------+---------------------+
  2. | id | k | c | TIDB_ROW_CHECKSUM() |
  3. +----+------+------+---------------------+
  4. | 1 | 10 | a | 3813955661 |
  5. +----+------+------+---------------------+
  6. 1 row in set (0.000 sec)

MySQL 兼容性

TIDB_ROW_CHECKSUM 是 TiDB 特有的函数,和 MySQL 不兼容。

另请参阅