您当前的位置:首页 > 电脑百科 > 数据库 > MYSQL

MySQL 8.0新特性之隐藏字段的深入讲解

时间:2021-07-12 10:53:39  来源:  作者:开心一刻成长每一天

这篇文章主要给大家介绍了关于MySQL 8.0新特性之隐藏字段的相关资料,文中通过示例代码介绍得非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

前言
MySQL 8.0.23 版本增加了一个新的功能:隐藏字段(Invisible Column),也称为不可见字段。本文给大家介绍一下 MySQL 隐藏字段的相关概念和具体实现。

基本概念
隐藏字段需要在查询中进行显式引用,否则对查询而言是不可见的。MySQL 8.0.23 开始支持隐藏字段,在此之前所有的字段都是可见字段。

考虑以下应用场景,假如一个应用程序使用SELECT 语句访问某个表,并且必须持续不断地进行查询,即使我们为该表增加了一个该应用不需要的新字段时也要求能够正常工作。对于SELECT 查询,星号()代表了表中除隐藏字段之外的所有字段,因此我们可以将新加的字段定义为隐藏字段。该隐藏字段对于SELECT 查询是不可见的,因此应用能够继续运行。如果新版本的应用程序需要使用该字段,可以在查询中显式指定。

PS:不推荐使用SELECT *语句查询数据,应该明确指定需要返回的字段。

隐藏字段与 DDL 语句
默认情况下创建的字段属于可见字段。如果想要显式指定字段的可见性,可以在CREATE TABLE或者ALTER TABLE语句中为字段的定义指定VISIBLE 或者 INVISIBLE 关键字。例如:

CREATE TABLE t1 (
i INT,
j DATE INVISIBLE
) ENGINE = InnoDB;
ALTER TABLE t1 ADD COLUMN k INT INVISIBLE;
ALTER TABLE t1 CHANGE COLUMN j j DATE VISIBLE;
ALTER TABLE t1 MODIFY COLUMN j DATE INVISIBLE;
ALTER TABLE t1 ALTER COLUMN j SET VISIBLE;

使用隐藏字段时,需要注意以下事项:

  • 一个表至少需要一个可见字段。如果将所有字段都设置为隐藏字段,将会返回错误。
  • 隐藏字段支持常见的字段属性:NULL、NOT NULL 以及 AUTO_INCREMENT 等等。
  • 计算列(Generated column)可以是隐藏字段。
  • 索引可以使用隐藏字段,包括 PRIMARY KEY 和 UNIQUE 索引。虽然一个表至少需要一个可见字段,但是索引定义中可以不包含任何可见字段。
  • 删除某个表中的隐藏字段时,同时会从相关索引中删除该字段。
  • 外键约束可以基于隐藏字段进行定义,同时外键约束也可以引用隐藏字段。
  • CHECK 约束可以基于隐藏字段进行定义。插入或者更新数据时,如果违反了隐藏字段上的 CHECK 约束将会返回错误。

如果使用CREATE TABLE ... LIKE语句复制表结构,将会复制原表中的隐藏字段,而且它们在新表中仍然是隐藏字段。

如果使用CREATE TABLE ... SELECT语句复制表,不会包含隐藏字段,除非显式指定了隐藏字段。尽管如此,即使包含了原表中的隐藏字段,新表中的这些字段将会变成可见字段。例如:

mysql> CREATE TABLE t1 (col1 INT, col2 INT INVISIBLE);
mysql> CREATE TABLE t2 AS SELECT col1, col2 FROM t1;
mysql> SHOW CREATE TABLE t2G
*************************** 1. row ***************************
Table: t2
Create Table: CREATE TABLE t2 (
col1 int DEFAULT NULL,
col2 int DEFAULT NULL
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci

如果想要保留这些字段的隐藏属性,可以在 CREATE TABLE 之后为它们指定隐藏属性。例如:

mysql> CREATE TABLE t1 (col1 INT, col2 INT INVISIBLE);
mysql> CREATE TABLE t2 (col2 INT INVISIBLE) AS SELECT col1, col2 FROM t1;
mysql> SHOW CREATE TABLE t2G
*************************** 1. row ***************************
Table: t2
Create Table: CREATE TABLE t2 (
col1 int DEFAULT NULL,
col2 int DEFAULT NULL /!80023 INVISIBLE /
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci

视图可以引用隐藏字段,需要在定义中显式指定这些字段。在视图定义之后修改字段的可见性不会影响视图。

隐藏字段与 DML 语句
对于 SELECT 语句,除非在查询列表中显式指定了隐藏字段,否则查询结构中不会包含隐藏字段。查询列表中的 和 tbl_name. 不会包含隐藏字段。自然连接不会包含隐藏字段。

对于以下语句:

mysql> CREATE TABLE t1 (col1 INT, col2 INT INVISIBLE);
mysql> INSERT INTO t1 (col1, col2) VALUES(1, 2), (3, 4);

mysql> SELECT * FROM t1;
+------+
| col1 |
+------+
| 1 |
| 3 |
+------+

mysql> SELECT col1, col2 FROM t1;
+------+------+
| col1 | col2 |
+------+------+
| 1 | 2 |
| 3 | 4 |
+------+------+

第一个 SELECT 语句没有引用隐藏字段 col2(* 不包含隐藏字段),因此查询结果中没有返回 col2 字段。第二个 SELECT 语句显式指定了 col2 字段,因此查询结果返回了该字段。

对于查询语句,如果没有为隐藏字段指定数据,使用隐式默认值规则进行赋值。

对于 INSERT 语句(包括 REPLACE 语句的数据插入),如果没有指定字段列表、指定空白列表或者没有在字段列表中指定隐藏字段时,使用隐式默认值赋值。例如:

CREATE TABLE t1 (col1 INT, col2 INT INVISIBLE);
INSERT INTO t1 VALUES(...);
INSERT INTO t1 () VALUES(...);
INSERT INTO t1 (col1) VALUES(...);

对于前两个 INSERT 语句,VALUES() 列表必须为每个可见字段和隐藏字段提供一个数值。对于第三个 INSERT 语句,VALUES() 列表必须为每个指定的字段提供一个数值。

对于 LOAD DATA 和 LOAD XML 语句,如果没有指定字段列表或者没有在字段列表中指定隐藏字段时,使用隐式默认值赋值。输入数据中不能包含隐藏字段的值。如果想要为上面的语句提供一个非默认的数据,可以在字段列表中显式指定隐藏字段并且在 VALUES() 列表中指定一个数值。

INSERT INTO … SELECT 和 REPLACE INTO … SELECT 不会包含隐藏字段,因为 * 不会返回隐藏字段。此时同样会使用隐式默认值规则进行赋值。

基于 PRIMARY KEY 或者 UNIQUE 索引执行插入或者忽略插入、替换或者修改数据的语句中,MySQL 对隐藏字段的处理方式和可见字段相同:隐藏字段同样会用于键值得比较。准确来说,如果某个新的数据行和已有数据行的唯一键字段值相同,无论索引字段是否可见,都会使用以下处理方式:

如果指定了 IGNORE 修饰符,INSERT、LOAD DATA 以及 LOAD XML 都会忽略新的数据行。
REPLACE 使用新的数据行替换原有的数据行。如果指定了 REPLACE 修饰符,LOAD DATA 和 LOAD XML 也是如此。
INSERT … ON DUPLICATE KEY UPDATE 更新原有的数据行。
如果想要使用 UPDATE 语句更新隐藏字段,像可见字段一样显式进行赋值即可。

隐藏字段相关的元数据
我们可以通过
INFORMATION_SCHEMA.COLUMNS 系统表的 EXTRA 字段或者 SHOW COLUMNS 命令查看字段的可见属性。例如:

mysql> SELECT TABLE_NAME, COLUMN_NAME, EXTRA
FROM INFORMATION_SCHEMA.COLUMNS
WHERE TABLE_SCHEMA = 'test' AND TABLE_NAME = 't1';
+------------+-------------+-----------+
| TABLE_NAME | COLUMN_NAME | EXTRA  |
+------------+-------------+-----------+
| t1   | i   |   |
| t1   | j   |   |
| t1   | k   | INVISIBLE |
+------------+-------------+-----------+


默认情况下字段是可见的,此时 EXTRA 字段为空。对于隐藏字段,EXTRA 显式为 INVISIBLE。

SHOW CREATE TABLE 命令可以显式表中的隐藏字段,字段定义中包含一个基于版本的注释,其中包含了一个 INVISIBLE 关键字:

mysql> SHOW CREATE TABLE t1G
*************************** 1. row *************************** 
Table: t1
Create Table: CREATE TABLE t1 (
i int DEFAULT NULL,
j int DEFAULT NULL,
k int DEFAULT NULL /!80023 INVISIBLE /
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci

mysqldump 和 mysqlpump 使用 SHOW CREATE TABLE 命令,因此它们导出的表定义中包含可隐藏字段。同时,它们在导出的数据中包含了隐藏字段的值。如果将导出文件加载到不支持的隐藏字段的低版本 MySQL 中,将会忽略基于版本的注释信息,从而将隐藏字段作为可见字段使用。

 

隐藏字段与二进制日志
对于二进制日志中的事件,MySQL 使用以下方式处理隐藏字段:

创建表的事件中包含了隐藏字段的 INVISIBLE 属性。
数据行事件中的隐藏字段和可见字段处理方式相同。它们会根据系统变量 binlog_row_image 的设置进行处理。
当数据行事件被应用时,隐藏字段和可见字段处理方式相同。其中,使用的算法和索引基于系统变量
slave_rows_search_algorithms 的设置进行选择。
计算写入集(writeset)时隐藏字段和可见字段处理方式相同。写入集中包含了基于隐藏字段定义的索引。
mysqlbinlog 命令中包含了字段元数据中的可见属性。



Tags:MySQL   点击:()  评论:()
声明:本站部分内容及图片来自互联网,转载是出于传递更多信息之目的,内容观点仅代表作者本人,如有任何标注错误或版权侵犯请与我们联系(Email:2595517585@qq.com),我们将及时更正、删除,谢谢。
▌相关推荐
满怀忧思不如先干再说!前言数据库拆分属于中高级开发要做的事情,不过具体的看企业吧。各种情况都会出现,这篇文章主要是基于Mycat去实现一下数据库拆分,至于拆分的思想,后边补一...【详细内容】
2022-11-01  Tags: MySQL  点击:(5)  评论:(0)  加入收藏
一. 前言这一篇来详细看看 EXPLAIN 各个参数的含义和扩展 , 整理出来便于使用时快速查询二 . explain 使用 三. 业务实践在日常实践中 , 我们应该如何使用 explain 提供的...【详细内容】
2022-10-17  Tags: MySQL  点击:(17)  评论:(0)  加入收藏
我的故事从一个问题开始,当时我发现了新的错误:[22001][1690] 数据截断:BIGINT UNSIGNED 值超出 '(`cur`.`count_rows` - `perv`.`count_rows`)' 的范围嗯,我明白发生...【详细内容】
2022-10-16  Tags: MySQL  点击:(25)  评论:(0)  加入收藏
写在前面 分享一些MySQL(MariaDB)集群主从结构数据读写分离的笔记,关于读写分离: 一如果对于读密集型应用,可以容忍从库异步复制延迟导致的脏数据,读写分离是一种不错的负载均...【详细内容】
2022-10-15  Tags: MySQL  点击:(22)  评论:(0)  加入收藏
从MySQL官方网站上看,MySQL整体分为了商业版本和开源版本两大类。一、商业版本1、MySQL HeatWave它是MySQL官方提供的云服务,直接部署在云上,运维更加简单,能够灵活进行扩缩容,能...【详细内容】
2022-10-09  Tags: MySQL  点击:(123)  评论:(0)  加入收藏
1. 问题:怎么给线上表加字段?工作中最常遇到的问题,怎么给线上频繁使用的大表添加字段?比如:给下面的用户表(user)添加年龄(age)字段。CREATE TABLE `user` ( `id` int NOT NULL AUT...【详细内容】
2022-10-08  Tags: MySQL  点击:(38)  评论:(0)  加入收藏
如何搭建IIS+MySQL+PHP环境教程方法首先下载php-5.2.0-win32.zip,mysql-noinstall-5.0.22-win32.zip和phpMyAdmin-2.9.1.1-all-languages.zip。这三个文件的下载地址可以在百...【详细内容】
2022-09-27  Tags: MySQL  点击:(51)  评论:(0)  加入收藏
1. 概览在过去的3个月中,我们发布上线了冷热数据分离存储等功能。今天很高兴和大家交流PolarDB-X最新的内核版本5.4.15。在最新版本中,提供诸多新的功能:存储过程,读写分离优...【详细内容】
2022-09-27  Tags: MySQL  点击:(30)  评论:(0)  加入收藏
作为CRUD程序员,大部分Java开发者应该都在用Mybatis Plus来操作数据库。但是BaseMapper默认仅提供了int insert(T entity)这个单条插入的方法。那么我们想批量插入数据该怎么...【详细内容】
2022-09-26  Tags: MySQL  点击:(98)  评论:(0)  加入收藏
B+树是一种在非叶子节点存放排序好的索引而在叶子节点存放数据的数据结构,值得注意的是,在叶子节点中,存储的并非只是一行表数据,而是以页为单位存储,一个页可以包含多行表记录。...【详细内容】
2022-09-26  Tags: MySQL  点击:(46)  评论:(0)  加入收藏
▌简易百科推荐
最近发现之前使用的MyBatis插件很久都没更新了,就想换个其他插件来用用。偶然发现MyBatis-Plus团队也开发了一款插件MyBatisX,体验了一把确实非常好用,提示很全,而且还能通过GUI...【详细内容】
2022-11-01  老诚不bug  今日头条  Tags:MyBatis-Plus   点击:(6)  评论:(0)  加入收藏
满怀忧思不如先干再说!前言数据库拆分属于中高级开发要做的事情,不过具体的看企业吧。各种情况都会出现,这篇文章主要是基于Mycat去实现一下数据库拆分,至于拆分的思想,后边补一...【详细内容】
2022-11-01  添甄  今日头条  Tags:MySQL   点击:(5)  评论:(0)  加入收藏
一. 前言这一篇来详细看看 EXPLAIN 各个参数的含义和扩展 , 整理出来便于使用时快速查询二 . explain 使用 三. 业务实践在日常实践中 , 我们应该如何使用 explain 提供的...【详细内容】
2022-10-17  架构师之道  今日头条  Tags:explain   点击:(17)  评论:(0)  加入收藏
我的故事从一个问题开始,当时我发现了新的错误:[22001][1690] 数据截断:BIGINT UNSIGNED 值超出 '(`cur`.`count_rows` - `perv`.`count_rows`)' 的范围嗯,我明白发生...【详细内容】
2022-10-16  庄志炎  今日头条  Tags:MySQL   点击:(25)  评论:(0)  加入收藏
写在前面 分享一些MySQL(MariaDB)集群主从结构数据读写分离的笔记,关于读写分离: 一如果对于读密集型应用,可以容忍从库异步复制延迟导致的脏数据,读写分离是一种不错的负载均...【详细内容】
2022-10-15  互联网资讯看板   网易号  Tags:读写分离   点击:(22)  评论:(0)  加入收藏
一,背景需求我们在实施虚拟化或者超融合的时候,部署的Linux业务虚拟机,因业务需求,需要对几个网卡做绑定。有些工程师可能不光从事安全工作,也从事主机存储配置类的工作,那么配置...【详细内容】
2022-10-14  互联共商  网易号  Tags:Linux网卡   点击:(17)  评论:(0)  加入收藏
从MySQL官方网站上看,MySQL整体分为了商业版本和开源版本两大类。一、商业版本1、MySQL HeatWave它是MySQL官方提供的云服务,直接部署在云上,运维更加简单,能够灵活进行扩缩容,能...【详细内容】
2022-10-09  鹿先生科技  今日头条  Tags:MySQL   点击:(123)  评论:(0)  加入收藏
1. 问题:怎么给线上表加字段?工作中最常遇到的问题,怎么给线上频繁使用的大表添加字段?比如:给下面的用户表(user)添加年龄(age)字段。CREATE TABLE `user` ( `id` int NOT NULL AUT...【详细内容】
2022-10-08  一灯架构  今日头条  Tags:MySQL   点击:(38)  评论:(0)  加入收藏
B+树是一种在非叶子节点存放排序好的索引而在叶子节点存放数据的数据结构,值得注意的是,在叶子节点中,存储的并非只是一行表数据,而是以页为单位存储,一个页可以包含多行表记录。...【详细内容】
2022-09-26  Java大数据高级架构师  今日头条  Tags:InnoDB   点击:(46)  评论:(0)  加入收藏
一、概述mysql是一个关系型数据库管理系统,由瑞典MySQL AB 公司开发,属于 Oracle 旗下产品。MySQL 是最流行的关系型数据库管理系统之一,在 WEB 应用方面,MySQL是最好的 RDBMS (...【详细内容】
2022-09-23  大数据老司机    Tags:MySQL   点击:(40)  评论:(0)  加入收藏
站内最新
站内热门
站内头条