约束

TiDB 支持的约束与 MySQL 的基本相同。

非空约束

TiDB 支持的非空约束规则与 MySQL 支持的一致。例如:

  1. CREATE TABLE users (
  2. id INT NOT NULL PRIMARY KEY AUTO_INCREMENT,
  3. age INT NOT NULL,
  4. last_login TIMESTAMP
  5. );
  1. INSERT INTO users (id,age,last_login) VALUES (NULL,123,NOW());
  1. Query OK, 1 row affected (0.02 sec)
  1. INSERT INTO users (id,age,last_login) VALUES (NULL,NULL,NOW());
  1. ERROR 1048 (23000): Column 'age' cannot be null
  1. INSERT INTO users (id,age,last_login) VALUES (NULL,123,NULL);
  1. Query OK, 1 row affected (0.03 sec)
  • 第一条 INSERT 语句成功,因为对于定义为 AUTO_INCREMENT 的列,允许 NULL 作为其特殊值。TiDB 将为其分配下一个自动值。

  • 第二条 INSERT 语句失败,因为 age 列被定义为 NOT NULL

  • 第三条 INSERT 语句成功,因为 last_login 列没有被明确地指定为 NOT NULL。默认允许 NULL 值。

CHECK 约束

TiDB 会解析并忽略 CHECK 约束。该行为与 MySQL 5.7 的相兼容。

示例如下:

  1. DROP TABLE IF EXISTS users;
  2. CREATE TABLE users (
  3. id INT NOT NULL PRIMARY KEY AUTO_INCREMENT,
  4. username VARCHAR(60) NOT NULL,
  5. UNIQUE KEY (username),
  6. CONSTRAINT min_username_length CHECK (CHARACTER_LENGTH(username) >=4)
  7. );
  8. INSERT INTO users (username) VALUES ('a');
  9. SELECT * FROM users;

唯一约束

唯一约束是指唯一索引和主键列中所有的非空值都是唯一的。

乐观事务

在 TiDB 的乐观事务中,默认会对唯一约束进行惰性检查。通过在事务提交时再进行批量检查,TiDB 能够减少网络开销、提升性能。例如:

  1. DROP TABLE IF EXISTS users;
  2. CREATE TABLE users (
  3. id INT NOT NULL PRIMARY KEY AUTO_INCREMENT,
  4. username VARCHAR(60) NOT NULL,
  5. UNIQUE KEY (username)
  6. );
  7. INSERT INTO users (username) VALUES ('dave'), ('sarah'), ('bill');

乐观事务模式下且 tidb_constraint_check_in_place=OFF

  1. BEGIN OPTIMISTIC;
  2. INSERT INTO users (username) VALUES ('jane'), ('chris'), ('bill');
  1. Query OK, 3 rows affected (0.00 sec)
  2. Records: 3 Duplicates: 0 Warnings: 0
  1. INSERT INTO users (username) VALUES ('steve'),('elizabeth');
  1. Query OK, 2 rows affected (0.00 sec)
  2. Records: 2 Duplicates: 0 Warnings: 0
  1. COMMIT;
  1. ERROR 1062 (23000): Duplicate entry 'bill' for key 'users.username'

在以上乐观事务的示例中,唯一约束的检查推迟到事务提交时才进行。由于 bill 值已经存在,这一行为导致了重复键错误。

你可通过设置 tidb_constraint_check_in_placeON 停用此行为(该变量仅适用于乐观事务,悲观事务需通过 tidb_constraint_check_in_place_pessimistic 设置)。当 tidb_constraint_check_in_place 设置为 ON 时,TiDB 会在执行语句时就对唯一约束进行检查。例如:

  1. DROP TABLE IF EXISTS users;
  2. CREATE TABLE users (
  3. id INT NOT NULL PRIMARY KEY AUTO_INCREMENT,
  4. username VARCHAR(60) NOT NULL,
  5. UNIQUE KEY (username)
  6. );
  7. INSERT INTO users (username) VALUES ('dave'), ('sarah'), ('bill');
  1. SET tidb_constraint_check_in_place = ON;
  1. Query OK, 0 rows affected (0.00 sec)
  1. BEGIN OPTIMISTIC;
  1. Query OK, 0 rows affected (0.00 sec)
  1. INSERT INTO users (username) VALUES ('jane'), ('chris'), ('bill');
  1. ERROR 1062 (23000): Duplicate entry 'bill' for key 'users.username'

第一条 INSERT 语句导致了重复键错误。这会造成额外的网络通信开销,并可能降低插入操作的吞吐量。

悲观事务

在 TiDB 的悲观事务中,默认在执行任何一条需要插入或更新唯一索引的 SQL 语句时都会进行唯一约束检查:

  1. DROP TABLE IF EXISTS users;
  2. CREATE TABLE users (
  3. id INT NOT NULL PRIMARY KEY AUTO_INCREMENT,
  4. username VARCHAR(60) NOT NULL,
  5. UNIQUE KEY (username)
  6. );
  7. INSERT INTO users (username) VALUES ('dave'), ('sarah'), ('bill');
  8. BEGIN PESSIMISTIC;
  9. INSERT INTO users (username) VALUES ('jane'), ('chris'), ('bill');
  1. ERROR 1062 (23000): Duplicate entry 'bill' for key 'users.username'

对于悲观事务,你可以设置变量 tidb_constraint_check_in_place_pessimisticOFF 来推迟唯一约束检查,到下一次对该唯一索引项加锁时或事务提交时再进行检查,同时也跳过对该悲观锁加锁,以获得更好的性能。此时需要注意:

  • 由于推迟了唯一约束检查,TiDB 可能会读取到不满足唯一约束的结果,执行 COMMIT 语句时可能返回 Duplicate entry 错误。返回该错误时,TiDB 会回滚当前事务。

    下面这个例子跳过了对 bill 的加锁,因此 TiDB 可能读到不满足唯一性约束的结果:

    1. SET tidb_constraint_check_in_place_pessimistic = OFF;
    2. BEGIN PESSIMISTIC;
    3. INSERT INTO users (username) VALUES ('jane'), ('chris'), ('bill'); -- Query OK, 3 rows affected
    4. SELECT * FROM users FOR UPDATE;

    TiDB 读到了不满足唯一性约束的结果:有两个 bill

    1. +----+----------+
    2. | id | username |
    3. +----+----------+
    4. | 1 | dave |
    5. | 2 | sarah |
    6. | 3 | bill |
    7. | 7 | jane |
    8. | 8 | chris |
    9. | 9 | bill |
    10. +----+----------+

    此时,如果提交事务,TiDB 将进行唯一约束检查,报出 Duplicate entry 错误并回滚事务。

    1. COMMIT;
    1. ERROR 1062 (23000): Duplicate entry 'bill' for key 'users.username'
  • 关闭该变量时,如果在事务中写入数据,执行 COMMIT 语句可能会返回 Write conflict 错误。返回该错误时,TiDB 会回滚当前事务。

    在下面这个例子中,当有并发事务写入时,跳过悲观锁导致事务提交时报出 Write conflict 错误并回滚。

    1. DROP TABLE IF EXISTS users;
    2. CREATE TABLE users (
    3. id INT NOT NULL PRIMARY KEY AUTO_INCREMENT,
    4. username VARCHAR(60) NOT NULL,
    5. UNIQUE KEY (username)
    6. );
    7. SET tidb_constraint_check_in_place_pessimistic = OFF;
    8. BEGIN PESSIMISTIC;
    9. INSERT INTO users (username) VALUES ('jane'), ('chris'), ('bill'); -- Query OK, 3 rows affected

    然后另一个会话中写入了 bill

    1. INSERT INTO users (username) VALUES ('bill'); -- Query OK, 1 row affected

    在第一个会话中提交时,TiDB 会报出 Write conflict 错误。

    1. COMMIT;
    1. ERROR 9007 (HY000): Write conflict, txnStartTS=435688780611190794, conflictStartTS=435688783311536129, conflictCommitTS=435688783311536130, key={tableID=74, indexID=1, indexValues={bill, }} primary={tableID=74, indexID=1, indexValues={bill, }}, reason=LazyUniquenessCheck [try again later]
  • 关闭该变量时,如果多个悲观事务之间存在写冲突,悲观锁可能会在其它悲观事务提交时被强制回滚,因此产生 PessimisticLockNotFound 错误。发生该错误时,说明该业务不适合推迟悲观事务的唯一约束检查,应考虑调整业务避免冲突,或在发生错误后重试事务。

  • 关闭该变量会导致悲观事务中可能报出错误 8147: LazyUniquenessCheckFailure

    约束 - 图1

    注意

    返回 8147 错误时当前事务回滚。

    下面的例子在 INSERT 语句执行时跳过了一次加锁后,在 DELETE 语句执行时对该唯一索引加锁并检查,即会在该语句报错:

    1. SET tidb_constraint_check_in_place_pessimistic = OFF;
    2. BEGIN PESSIMISTIC;
    3. INSERT INTO users (username) VALUES ('jane'), ('chris'), ('bill'); -- Query OK, 3 rows affected
    4. DELETE FROM users where username = 'bill';
    1. ERROR 8147 (23000): transaction aborted because lazy uniqueness check is enabled and an error occurred: [kv:1062]Duplicate entry 'bill' for key 'users.username'
  • 关闭该变量时,1062 Duplicate entry 报错不一定是当前执行的 SQL 语句所发生的错误。因此,在一个事务操作多个表,且这些表有同名索引时,请注意 1062 报错信息中提示的是哪个表的哪个索引发生了错误。

主键约束

与 MySQL 行为一样,主键约束包含了唯一约束,即创建了主键约束相当于拥有了唯一约束。此外,TiDB 其他的主键约束规则也与 MySQL 相似。例如:

  1. CREATE TABLE t1 (a INT NOT NULL PRIMARY KEY);
  1. Query OK, 0 rows affected (0.12 sec)
  1. CREATE TABLE t2 (a INT NULL PRIMARY KEY);
  1. ERROR 1171 (42000): All parts of a PRIMARY KEY must be NOT NULL; if you need NULL in a key, use UNIQUE instead
  1. CREATE TABLE t3 (a INT NOT NULL PRIMARY KEY, b INT NOT NULL PRIMARY KEY);
  1. ERROR 1068 (42000): Multiple primary key defined
  1. CREATE TABLE t4 (a INT NOT NULL, b INT NOT NULL, PRIMARY KEY (a,b));
  1. Query OK, 0 rows affected (0.10 sec)

分析:

  • t2 创建失败,因为定义为主键的列 a 不能允许 NULL 值。
  • t3 创建失败,因为一张表只能有一个主键。
  • t4 创建成功,因为虽然只能有一个主键,但 TiDB 支持定义一个多列组合作为复合主键。

除上述规则外,TiDB 目前仅支持对 NONCLUSTERED 的主键进行添加和删除操作。例如:

  1. CREATE TABLE t5 (a INT NOT NULL, b INT NOT NULL, PRIMARY KEY (a,b) CLUSTERED);
  2. ALTER TABLE t5 DROP PRIMARY KEY;
  1. ERROR 8200 (HY000): Unsupported drop primary key when the table is using clustered index
  1. CREATE TABLE t5 (a INT NOT NULL, b INT NOT NULL, PRIMARY KEY (a,b) NONCLUSTERED);
  2. ALTER TABLE t5 DROP PRIMARY KEY;
  1. Query OK, 0 rows affected (0.10 sec)

要了解关于 CLUSTERED 主键的详细信息,请参考聚簇索引

外键约束

约束 - 图2

注意

TiDB 从 v6.6.0 开始支持外键约束。在 v6.6.0 之前,TiDB 支持创建和删除外键约束,但外键约束并不生效。升级到 v6.6.0 后,可以先删除不生效的外键后再创建外键使外键约束生效。

TiDB 支持创建外键约束。例如:

  1. CREATE TABLE users (
  2. id INT NOT NULL PRIMARY KEY AUTO_INCREMENT,
  3. doc JSON
  4. );
  5. CREATE TABLE orders (
  6. id INT NOT NULL PRIMARY KEY AUTO_INCREMENT,
  7. user_id INT NOT NULL,
  8. doc JSON,
  9. FOREIGN KEY fk_user_id (user_id) REFERENCES users(id)
  10. );
  1. SELECT table_name, column_name, constraint_name, referenced_table_name, referenced_column_name
  2. FROM information_schema.key_column_usage WHERE table_name IN ('users', 'orders');
  1. +------------+-------------+-----------------+-----------------------+------------------------+
  2. | table_name | column_name | constraint_name | referenced_table_name | referenced_column_name |
  3. +------------+-------------+-----------------+-----------------------+------------------------+
  4. | users | id | PRIMARY | NULL | NULL |
  5. | orders | id | PRIMARY | NULL | NULL |
  6. | orders | user_id | fk_user_id | users | id |
  7. +------------+-------------+-----------------+-----------------------+------------------------+
  8. 3 rows in set (0.00 sec)

TiDB 也支持使用 ALTER TABLE 命令来删除外键 (DROP FOREIGN KEY) 和添加外键 (ADD FOREIGN KEY):

  1. ALTER TABLE orders DROP FOREIGN KEY fk_user_id;
  2. ALTER TABLE orders ADD FOREIGN KEY fk_user_id (user_id) REFERENCES users(id);