MySQL 8.0新特性之隐藏字段的深入讲解
前言
MySQL8.0.23版本增加了一个新的功能:隐藏字段(InvisibleColumn),也称为不可见字段。本文给大家介绍一下MySQL隐藏字段的相关概念和具体实现。
基本概念
隐藏字段需要在查询中进行显式引用,否则对查询而言是不可见的。MySQL8.0.23开始支持隐藏字段,在此之前所有的字段都是可见字段。
考虑以下应用场景,假如一个应用程序使用SELECT*语句访问某个表,并且必需持续不断地进行查询,即使我们为该表增加了一个该应用不需要的新字段时也要求能够正常工作。对于SELECT*查询,星号(*)代表了表中除隐藏字段之外的所有字段,因此我们可以将新加的字段定义为隐藏字段。该隐藏字段对于SELECT*查询是不可见的,因此应用能够继续运行。如果新版本的应用程序需要使用该字段,可以在查询中显式指定。
PS:不推荐使用SELECT*语句查询数据,应该明确指定需要返回的字段。
隐藏字段与DDL语句
默认情况下创建的字段属于可见字段。如果想要显式指定字段的可见性,可以在CREATETABLE或者ALTERTABLE语句中为字段的定义指定VISIBLE或者INVISIBLE关键字。例如:
CREATETABLEt1( iINT, jDATEINVISIBLE )ENGINE=InnoDB; ALTERTABLEt1ADDCOLUMNkINTINVISIBLE;
如果想要修改某个字段的可见性,同样可以使用VISIBLE或者INVISIBLE关键字。例如:
ALTERTABLEt1CHANGECOLUMNjjDATEVISIBLE; ALTERTABLEt1MODIFYCOLUMNjDATEINVISIBLE; ALTERTABLEt1ALTERCOLUMNjSETVISIBLE;
使用隐藏字段时,需要注意以下事项:
- 一个表至少需要一个可见字段。如果将所有字段都设置为隐藏字段,将会返回错误。
- 隐藏字段支持常见的字段属性:NULL、NOTNULL以及AUTO_INCREMENT等等。
- 计算列(Generatedcolumn)可以是隐藏字段。
- 索引可以使用隐藏字段,包括PRIMARYKEY和UNIQUE索引。虽然一个表至少需要一个可见字段,但是索引定义中可以不包含任何可见字段。
- 删除某个表中的隐藏字段时,同时会从相关索引中删除该字段。
- 外键约束可以基于隐藏字段进行定义,同时外键约束也可以引用隐藏字段。
- CHECK约束可以基于隐藏字段进行定义。插入或者更新数据时,如果违反了隐藏字段上的CHECK约束将会返回错误。
如果使用CREATETABLE...LIKE语句复制表结构,将会复制原表中的隐藏字段,而且它们在新表中仍然是隐藏字段。如果使用CREATETABLE...SELECT语句复制表,不会包含隐藏字段,除非显式指定了隐藏字段。尽管如此,即使包含了原表中的隐藏字段,新表中的这些字段将会变成可见字段。例如:
mysql>CREATETABLEt1(col1INT,col2INTINVISIBLE); mysql>CREATETABLEt2ASSELECTcol1,col2FROMt1; mysql>SHOWCREATETABLEt2\G ***************************1.row*************************** Table:t2 CreateTable:CREATETABLE`t2`( `col1`intDEFAULTNULL, `col2`intDEFAULTNULL )ENGINE=InnoDBDEFAULTCHARSET=utf8mb4COLLATE=utf8mb4_0900_ai_ci
如果想要保留这些字段的隐藏属性,可以在CREATETABLE之后为它们指定隐藏属性。例如:
mysql>CREATETABLEt1(col1INT,col2INTINVISIBLE); mysql>CREATETABLEt2(col2INTINVISIBLE)ASSELECTcol1,col2FROMt1; mysql>SHOWCREATETABLEt2\G ***************************1.row*************************** Table:t2 CreateTable:CREATETABLE`t2`( `col1`intDEFAULTNULL, `col2`intDEFAULTNULL/*!80023INVISIBLE*/ )ENGINE=InnoDBDEFAULTCHARSET=utf8mb4COLLATE=utf8mb4_0900_ai_ci
视图可以引用隐藏字段,需要在定义中显式指定这些字段。在视图定义之后修改字段的可见性不会影响视图。
隐藏字段与DML语句
对于SELECT语句,除非在查询列表中显式指定了隐藏字段,否则查询结构中不会包含隐藏字段。查询列表中的*和tbl_name.*不会包含隐藏字段。自然连接不会包含隐藏字段。
对于以下语句:
mysql>CREATETABLEt1(col1INT,col2INTINVISIBLE); mysql>INSERTINTOt1(col1,col2)VALUES(1,2),(3,4); mysql>SELECT*FROMt1; +------+ |col1| +------+ |1| |3| +------+ mysql>SELECTcol1,col2FROMt1; +------+------+ |col1|col2| +------+------+ |1|2| |3|4| +------+------+
第一个SELECT语句没有引用隐藏字段col2(*不包含隐藏字段),因此查询结果中没有返回col2字段。第二个SELECT语句显式指定了col2字段,因此查询结果返回了该字段。
对于查询语句,如果没有为隐藏字段指定数据,使用隐式默认值规则进行赋值。
对于INSERT语句(包括REPLACE语句的数据插入),如果没有指定字段列表、指定空白列表或者没有在字段列表中指定隐藏字段时,使用隐式默认值赋值。例如:
CREATETABLEt1(col1INT,col2INTINVISIBLE); INSERTINTOt1VALUES(...); INSERTINTOt1()VALUES(...); INSERTINTOt1(col1)VALUES(...);
对于前两个INSERT语句,VALUES()列表必须为每个可见字段和隐藏字段提供一个数值。对于第三个INSERT语句,VALUES()列表必须为每个指定的字段提供一个数值。
对于LOADDATA和LOADXML语句,如果没有指定字段列表或者没有在字段列表中指定隐藏字段时,使用隐式默认值赋值。输入数据中不能包含隐藏字段的值。
如果想要为上面的语句提供一个非默认的数据,可以在字段列表中显式指定隐藏字段并且在VALUES()列表中指定一个数值。
INSERTINTO…SELECT*和REPLACEINTO…SELECT*不会包含隐藏字段,因为*不会返回隐藏字段。此时同样会使用隐式默认值规则进行赋值。
基于PRIMARYKEY或者UNIQUE索引执行插入或者忽略插入、替换或者修改数据的语句中,MySQL对隐藏字段的处理方式和可见字段相同:隐藏字段同样会用于键值的比较。准确来说,如果某个新的数据行和已有数据行的唯一键字段值相同,无论索引字段是否可见,都会使用以下处理方式:
- 如果指定了IGNORE修饰符,INSERT、LOADDATA以及LOADXML都会忽略新的数据行。
- REPLACE使用新的数据行替换原有的数据行。如果指定了REPLACE修饰符,LOADDATA和LOADXML也是如此。
- INSERT…ONDUPLICATEKEYUPDATE更新原有的数据行。
如果想要使用UPDATE语句更新隐藏字段,像可见字段一样显式进行赋值即可。
隐藏字段相关的元数据
我们可以通过INFORMATION_SCHEMA.COLUMNS系统表的EXTRA字段或者SHOWCOLUMNS命令查看字段的可见属性。例如:
mysql>SELECTTABLE_NAME,COLUMN_NAME,EXTRA FROMINFORMATION_SCHEMA.COLUMNS WHERETABLE_SCHEMA='test'ANDTABLE_NAME='t1'; +------------+-------------+-----------+ |TABLE_NAME|COLUMN_NAME|EXTRA| +------------+-------------+-----------+ |t1|i|| |t1|j|| |t1|k|INVISIBLE| +------------+-------------+-----------+
默认情况下字段是可见的,此时EXTRA字段为空。对于隐藏字段,EXTRA显式为INVISIBLE。
SHOWCREATETABLE命令可以显式表中的隐藏字段,字段定义中包含一个基于版本的注释,其中包含了一个INVISIBLE关键字:
mysql>SHOWCREATETABLEt1\G ***************************1.row*************************** Table:t1 CreateTable:CREATETABLE`t1`( `i`intDEFAULTNULL, `j`intDEFAULTNULL, `k`intDEFAULTNULL/*!80023INVISIBLE*/ )ENGINE=InnoDBDEFAULTCHARSET=utf8mb4COLLATE=utf8mb4_0900_ai_ci
mysqldump和mysqlpump使用SHOWCREATETABLE命令,因此它们导出的表定义中包含可隐藏字段。同时,它们在导出的数据中包含了隐藏字段的值。如果将导出文件加载到不支持的隐藏字段的低版本MySQL中,将会忽略基于版本的注释信息,从而将隐藏字段作为可见字段使用。
隐藏字段与二进制日志
对于二进制日志中的事件,MySQL使用以下方式处理隐藏字段:
- 创建表的事件中包含了隐藏字段的INVISIBLE属性。
- 数据行事件中的隐藏字段和可见字段处理方式相同。它们会根据系统变量binlog_row_image的设置进行处理。
- 当数据行事件被应用时,隐藏字段和可见字段处理方式相同。其中,使用的算法和索引基于系统变量slave_rows_search_algorithms的设置进行选择。
- 计算写入集(writeset)时隐藏字段和可见字段处理方式相同。写入集中包含了基于隐藏字段定义的索引。
- mysqlbinlog命令中包含了字段元数据中的可见属性。
总结
到此这篇关于MySQL8.0新特性之隐藏字段的文章就介绍到这了,更多相关MySQL8.0隐藏字段内容请搜索毛票票以前的文章或继续浏览下面的相关文章希望大家以后多多支持毛票票!
声明:本文内容来源于网络,版权归原作者所有,内容由互联网用户自发贡献自行上传,本网站不拥有所有权,未作人工编辑处理,也不承担相关法律责任。如果您发现有涉嫌版权的内容,欢迎发送邮件至:czq8825#qq.com(发邮件时,请将#更换为@)进行举报,并提供相关证据,一经查实,本站将立刻删除涉嫌侵权内容。