之前的月报我们写过一篇关于 Window Function 的介绍,除此之外,10.2.2 又即将发布一些新的特性。

目前有两种类型的虚拟列:PERSISTENT/STORED 类型,这种类型的虚拟列的值是直接存在表中的;而 VIRTUAL 类型,其实只是一个定义,表结构中并不包括这个列,在需要用到的时候临时计算。默认值是 VIRTUAL。

虚拟列的语法

限制

  1. 只有支持的引擎才能使用虚拟列,目前InnoDB, Aria, MyISAM 和 CONNECT 引擎都能支持。
  2. 有限的支持虚拟列索引。VIRTUAL 类型的虚拟列不能作为主键也不能建索引。
  3. PERSISTENT 类型的虚拟列可以建索引,也可以作为外键,但是不支持 ON UPDATE CASCADE, ON UPDATE SET NULL, ON DELETE SET NULL。
  4. 不能用使用自定义函数来定义虚拟列,内置的函数可以。
  5. 不能定义虚拟列的虚拟列。

用例

表结构如下

用 DESCRIBE 语法来看表结构,在 Extra 列能看到相关的信息。

  1. DESCRIBE table1;
  2. +-------+-------------+------+-----+---------+------------+
  3. | Field | Type | Null | Key | Default | Extra |
  4. +-------+-------------+------+-----+---------+------------+
  5. | a | int(11) | NO | | NULL | |
  6. | b | varchar(32) | YES | | NULL | |
  7. | c | int(11) | YES | | NULL | VIRTUAL |
  8. | d | varchar(5) | YES | | NULL | PERSISTENT |
  9. +-------+-------------+------+-----+---------+------------+

再来做一些操作看效果

  1. INSERT INTO table1 VALUES (1, 'some text',default,default);
  2. Query OK, 1 row affected (0.00 sec)
  3. Query OK, 1 row affected, 1 warning (0.00 sec)
  4. Warning (Code 1645): The value specified for computed column 'c' in table 'table1' ignored.
  5. INSERT INTO table1 VALUES (123, 'even more text',default,'something');
  6. Query OK, 1 row affected, 2 warnings (0.00 sec)
  7. Warning (Code 1645): The value specified for computed column 'd' in table 'table1' ignored.
  8. SELECT * FROM table1;
  9. +-----+----------------+------+-------+
  10. | a | b | c | d |
  11. +-----+----------------+------+-------+
  12. | 1 | some text | 1 | some |
  13. | 2 | more text | 2 | more |
  14. | 123 | even more text | 3 | even |
  15. +-----+----------------+------+-------+
  16. 3 rows in set (0.00 sec)

在当前的所有MySQL版本及分支中,都只能定义固定的 DEFAULT 值,或者特定的类型如 AUTO_INCREMENT, CURRENT_TIMESTAMP。 在 10.1 中能支持 DATETIME 类型也使用 CURRENT_TIMESTAMP 定义默认值。 然而 MariaDB 10.2 中已经可以使用表达式来定义固定值。

例如你可以这么定义:

但是表达式不支持存储函数和子查询,而且表达式中如果包含另外的列,那么那个列必须在当前列之前定义了。

并且 BLOB 和 TEXT 类型也支持DEFAULT值定义了。

但是必须注意,在 DEFAULT 使用表达式,必须使用ROW格式才能保证复制的数据安全,当然,任何时候我们都建议使用ROW_FORMAT。

CHECK的语法

InnoDB是支持外键约束的,基本语法如下:

  1. [CONSTRAINT [symbol]] FOREIGN KEY
  2. [index_name] (index_col_name, ...)
  3. REFERENCES tbl_name (index_col_name,...)
  4. [ON DELETE reference_option]
  5. [ON UPDATE reference_option]
  6. RESTRICT | CASCADE | SET NULL | NO ACTION
  1. CHECK(expression) # 用于检查列的值,在列定义的时候使用
  2. CONSTRAINT [constraint_name] CHECK (expression)

用例

比如我们可以定义一张这样的表。

比如做一些最简单的数值检查

  1. CREATE TABLE t1 (a INT CHECK (a>2), b INT CHECK (b>2), CONSTRAINT a_greater CHECK (a>b));
  2. INSERT INTO t1(a) VALUES (1);
  3. ERROR 4022 (23000): CONSTRAINT `a` failed for `test`.`t1`
  4. INSERT INTO t1(a,b) VALUES (3,4);
  5. ERROR 4022 (23000): CONSTRAINT `a_greater` failed for `test`.`t1`
  6. INSERT INTO t1(a,b) VALUES (4,3);
  7. Query OK, 1 row affected (0.04 sec)

可以看到 CHECK 检查的条件生效了。

当然更复杂一点还能引入一些内置函数,包括UDF来做检查:

  1. CREATE TABLE t2 (name VARCHAR(30) CHECK (CHAR_LENGTH(name)>2), start_date DATE,
  2. end_date DATE CHECK (start_date IS NULL OR end_date IS NULL OR start_date<end_date));
  3. INSERT INTO t2(name, start_date, end_date) VALUES('Ione', '2003-12-15', '2014-11-09');
  4. Query OK, 1 row affected (0.04 sec)
  5. INSERT INTO t2(name, start_date, end_date) VALUES('Io', '2003-12-15', '2014-11-09');
  6. ERROR 4022 (23000): CONSTRAINT `name` failed for `test`.`t2`
  7. INSERT INTO t2(name, start_date, end_date) VALUES('Ione', NULL, '2014-11-09');
  8. Query OK, 1 row affected (0.04 sec)

这样在某些情况下可以把一些简单的逻辑检查放到数据库做,大大简化程序的设计。

MariaDB 10.2 是 MariaDB 一个较大的里程碑版本,很多社区呼声很大的功能都被扩充进去,对用户来说是个福音。MariaDB和MySQL相互竞争,对整个MySQL及其兼容分支的发展都是一个很好的现象,有竞争就有动力,无论是MySQL还是MariaDB,在强大的竞争压力下,进度都快起来了,新功能不断的加入,代码不断的优化,最终受益的都是我们这些用户。