MySQL 8.0 新特性之隐藏字段_mysql支持invisible列吗-程序员宅基地

技术标签: 不可见字段  MySQL  column  invisible  mysql  隐藏字段  

大家好,我是只谈技术不剪发的 Tony 老师。

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;

如果想要修改某个字段的可见性,同样可以使用 VISIBLE 或者 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 t2\G
*************************** 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 t2\G
*************************** 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 t1\G
*************************** 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 命令中包含了字段元数据中的可见属性。
版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/horses/article/details/113838642

智能推荐

【Tensorflow+Keras】tf.keras.layers.LSTM的解析与使用-程序员宅基地

文章浏览阅读2.1w次,点赞24次,收藏96次。1 作用原理实现LSTM网络原理介绍:长短期记忆模型(Long-Short Term Memory,LSTM)2 参数tf.keras.layers.LSTM(units,activation=“tanh”,recurrent_activation=“sigmoid”,#用于重复步骤的激活功能use_bias=True,#,是否图层使用偏置向量kernel_initializer=“glorot_uniform”,#kernel权重矩阵的 初始化程序,用于输入的线性转换recurren_tf.keras.layers.lstm

《操作系统实验》C++实现生产者-消费者问题_用线程模拟实现生产者和消费者的程序。缓冲区大小为5(可以定义数组来作为存储产品-程序员宅基地

文章浏览阅读1w次,点赞14次,收藏178次。生产者-消费者问题1 实验内容及要求2 实验环境3 实验设计3.1 问题描述3.2 基本思想3.2.1 生产者线程3.2.2 消费者线程3.2.3 同步的实现3.3 数据结构4 实验源码5 实验总结1 实验内容及要求1、模拟生产者—消费者问题;2、程序要添加适当的注释,程序的书写要采用缩进格式;3、程序要具在一定的健壮性,即当输入数据非法时,程序也能适当地做出反应;4、程序要做到界面友好,在程序运行时用户可以根据相应的提示信息进行操作。2 实验环境windows 10,Visual Studi_用线程模拟实现生产者和消费者的程序。缓冲区大小为5(可以定义数组来作为存储产品

#####Docker私有仓库的搭建和加认证####_registry-cli-程序员宅基地

文章浏览阅读328次。1.下载registry镜像[root@server1 docker]# docker pull registry:2: Pulling from library/registryc87736221ed0: Pull complete 1cc8e0bb44df: Pull complete 54d33bcb37f5: Pull complete e8afc091c171: Pull c..._registry-cli

java nio系列教程(2)---channel(通道介绍)和使用_messageutil.bytebuffertoarray-程序员宅基地

文章浏览阅读2.3k次。大家推荐个靠谱的公众号程序员探索之路,公众号内点击网赚获取彩蛋,大家一起加油 ​package com.zzh.buffer;import com.google.common.collect.Lists;import org.junit.jupiter.api.Test;import java.io.FileInputStream;import java.io.FileOut..._messageutil.bytebuffertoarray

官宣 | Apache Flink 1.12.0 正式发布,流批一体真正统一运行!_cannot resolve org.apache.flink:flink-web-socket:1-程序员宅基地

文章浏览阅读5.7k次,点赞9次,收藏10次。‍‍翻译 | 付典Review |徐榜江、朱翥Apache Flink 社区很荣幸地宣布 Flink 1.12.0 版本正式发布!近 300 位贡献者参与了 Flink 1.12.0 ..._cannot resolve org.apache.flink:flink-web-socket:1.12.0

计算机硬件技术的应用毕业论文,计算机硬件技术论文-程序员宅基地

文章浏览阅读528次。为毕业生写计算机硬件技术论文提供计算机硬件技术论文范文参考,涵盖硕士、大学本科毕业论文范文和职称论文范文,包括论文选题、开题报告、文献综述、任务书、参考文献等,是优秀免费计算机硬件技术论文网站。计算机硬件组装及维护技术的探讨摘要:随着社会进步与科技的快速发展,计算机的应用已经成为我们生活工作中的一个重要组成部分。因此,我们需要掌握一些基本的计算机硬件组装。计算机硬件维护保养技术探讨摘要】随着计算机..._计算机硬件技术理解与应用论文

随便推点

ucore lab7_ucorelab7-程序员宅基地

文章浏览阅读541次。练习1: 理解内核级信号量的实现和基于内核级信号量的哲学家就餐问题(不需要编码)比较Lab6和Lab7,发现区别主要在/kern/sync这个文件夹,其中就是实现信号量的一些代码。Baidu百科哲学家就餐问题可以这样表述,假设有五位哲学家围坐在一张圆形餐桌旁,做以下两件事情之一:吃饭,或者思考。吃东西的时候,他们就停止思考,思考的时候也停止吃东西。餐桌中间有一大碗意大利面,每两个哲学家之间有一只餐叉。因为用一只餐叉很难吃到意大利面,所以假设哲学家必须用两只餐叉吃东西。他们只能使用自己左右手边的那两_ucorelab7

数学表达式学习一_数学表达式i∈1到n-程序员宅基地

文章浏览阅读244次。学数学,吃透概念理论确实是基本的,但更要能精确表达出来,数学表达式,在此之前其实没什么接触,在闵老师的介绍之下,对论文的书写是必不可少,所以从陌生到熟悉到掌握,这个流程得一步一脚印,踏实走,走出舒适区,把文字转化成符号,转变心态,目前的问题就是不熟悉各字体的符号,相同字体的不同形态的使用,上下标什么时候标识等基本的,所以一边用一边学,照猫画虎。_数学表达式i∈1到n

java小白,基础薄弱,必看_java基础薄弱怎么办-程序员宅基地

文章浏览阅读159次。本教程涵盖了Java基础入门部分-基础进阶部分-项目实战部分,环环紧扣,循序渐进,是学习Java的不二路径,小白零基础都可以入手。基础开发环境开发工具:Eclipse(MyEclipse、idea、sts)我这里用的是Eclipse(你们应该也是用的这个,是吧?)Java版本:JDK 1.8 Tomcat7数据库:MySQL 5.7 数据库编译工具navicat一共三部分:基础入门(基础入门(以下二选一即可))入门视频教程1(精讲,耗时短,快速入门)48讲Java入门视频教程及课堂源码入门_java基础薄弱怎么办

We’re sorry but XXX doesn’t work properly without JavaScript enabled(解决方案汇总)_doesn't work properly without javascript enabled-程序员宅基地

文章浏览阅读10w+次,点赞16次,收藏25次。我遇到的问题:本地开发环境,访问接口状态为200,但没有正确但返回数据,返回信息提示为“We’re sorry but template doesn’t work properly without JavaScript enabled. Please enable it to continue.”网上搜索的解决方法没有解决问题,最终是在服务代理发现了问题,代理信息写错了,修改代理信息就改好了。以下汇总了其他的情况的解决方法1、mode类型前端修改方式:将mode类型由history改成hash;_doesn't work properly without javascript enabled

鸿蒙系统与应用场景,华为鸿蒙操作系统正式发布-程序员宅基地

文章浏览阅读627次。原标题:华为鸿蒙操作系统正式发布深圳湾现场报道,2019 年 8 月 9 日,华为全球开发者大会在东莞正式举行,会上,华为正式发布操作系统,这是一个基于微内核、面向全场景的分布式操作系统。随着华为全场景智慧生活战略的不断完善,鸿蒙 OS 将作为华为迎接全场景体验时代到来的产物,发挥其轻量化、小巧、功能强大的优势,率先应用在智能手表、智慧屏、车载设备、智能音箱等智能终端上,着力构建一个跨终端的融合共..._鸿蒙操作系统发布

HTNL5--<video>/<audio>随笔-程序员宅基地

文章浏览阅读174次。WHATWG 致力于 web 表单和应用程序,而 W3C 专注于 XHTML 2.0。在 2006 年,双方决定进行合作,来创建一个新版本的 HTML——HTML5。为 HTML5 建立的一些规则:新特性应该基于 HTML、CSS、DOM 以及 JavaScript。减少对外部插件的需求(比如 Flash)更优秀的错误处理更多取代脚本的标记HTML5 应该独..._var myvideo = document.getelementbyid("video");