浅析MySQL中的计算列(Generated Column列)与计算字段的介绍与应用-如何让数据库中某个字段随时间自动更新
一、计算列
MySQL 的 Generated Column 又称为虚拟列或计算列。Generated Column列的值是在列定义时包含了一个计算表达式计算得到的。
1、定义Generated column列的语法如下:
列名 类型
[GENERATED ALWAYS] AS (expr) [VIRTUAL | STORED]
[NOT NULL | NULL]
[UNIQUE [KEY]]
[[PRIMARY] KEY]
[COMMENT 'string']
(1)AS(expr)用于生成计算列值的表达式。
(2)VIRTUAL或STORED关键字表示是否存储计算列的值:
VIRTUAL:列值不存储,虚拟列不占用存储空间,默认设置为VIRTUAL。
STORED:在添加或更新行时计算并存储列值。存储列需要存储空间,并且可以创建索引。
2、Generated column 表达式必须遵循以下规则。如果表达式包含不允许的定义方式,则会发生错误。
(1)允许使用文本、内置函数和运算符,但不能使用返回值不确定的函数,比如NOW()。
(2)不允许使用存储函数和用户定义函数。
(3)不允许使用存储过程和函数参数。
(4)不允许使用变量(系统变量、用户定义变量和存储程序的局部变量)。
(5)不允许子查询。
(6)计算列在定义时可以引用其他的计算列,但只能引用表定义中较早出现的列。
(7)可以在计算列上创建索引,但不能在VIRTUAL类型的计算列上创建聚集索引。
3、计算列举例
(1)表的定义
mysql> create table sales(
-> goods_id int primary key,
-> goods_name char(20),
-> unit_price int,
-> quantity int,
-> amount int generated always as (unit_price*quantity));
Query OK, 0 rows affected (0.02 sec)
(2)插入数据:amount 自动为 8
mysql> insert into sales(goods_id,goods_name,unit_price,quantity) values(100101,'Apple',2,4);
Query OK, 1 row affected (0.00 sec)
mysql> select * from sales;
+----------+------------+------------+----------+--------+
| goods_id | goods_name | unit_price | quantity | amount |
+----------+------------+------------+----------+--------+
| 100101 | Apple | 2 | 4 | 8 |
+----------+------------+------------+----------+--------+
1 row in set (0.00 sec)
(3)查看创建表的语句:可见,计算列的默认类型为VIRTUAL。
mysql> show create table sales\G
*************************** 1. row ***************************
Table: sales
Create Table: CREATE TABLE `sales` (
`goods_id` int(11) NOT NULL,
`goods_name` char(20) DEFAULT NULL,
`unit_price` int(11) DEFAULT NULL,
`quantity` int(11) DEFAULT NULL,
`amount` int(11) GENERATED ALWAYS AS ((`unit_price` * `quantity`)) VIRTUAL,
PRIMARY KEY (`goods_id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4
1 row in set (0.00 sec)
4、计算列区别于需要我们手动或者程序给予赋值的列,它的值来源于该表中其它列的计算值。
比如,一个表中包含有数量列Number与单价列Price,我们就可以创建计算列金额Amount来表示数量*单价的结果值,创建Amount列后,在程序中需要使用计算金额这个值时,就不用取出Number列与Price列的值后相乘,而是直接取Amount列的值就可以了。
那么这个计算列要如何建立呢?先看通过sql的方法创建:
create table table1(
number decimal(18,4),
price money,
Amount as number*price --这里就是计算列
)
计算列是不需要我们指定数据类型与是否允许为null等信息的,SqlServer会根据情况自动赋予数据类型。在microsoft sql server management studio建创计算列更是简单。如下图所示,只要在列属性中的"计算所得的列规范"-"公式"中填写计算列的公式就可以了。
在上面的图片中我们也看到有"是持久的"这个选项,这个选项有什么用呢?
计算列如果没有特殊的设定,它将会是一个虚拟列,也就是这个列实际上是不存在的,只是每次要取这列的值时,sql会按照计算列的公式计算一次,再把结果返回给我们。
这样就会存在一些问题,比如,每次计算都会消耗一定的时间,而且也不能在这个列上创建索引。
那么能不能把计算列的结果存起来,每次取数据的时候直接把结果返回给我们,而不用每次去计算。创建计算列时把"是持久的"这个选项勾起来,就能达到我们的目的了,这时候,计算列就是一个实实在在的列,也可以在该列上创建索引了。
如果要查看所有已经存在计算列以及该计算列是否为"是持久的",可以利用sys.computed_columns视图
5、注意:
(1)计算列如果没有设置为"是持久的",那么它是不可以用来做check,foreign key或not null约束。当然,如果我们在microsoft sql server management studio为计算列设置了check等约束了,sql server会自动将该列设置为"是持久的"。
(2)计算列不可以再次用来作为中一个计算列的一部分。
(3)在触发器,不可以对计算列进行update判断,否则会报如下错误:列不能在IF UPDATE子句中使用,因为它是计算列。
二、SQL Server 中如何让数据库中某一个字段随时间自动更新?
今天在sql server论坛看到一个帖子:如何让数据库中某一个字段随时间自动更新?
那么如何来实现呢?可以用触发器,那么先要写个触发器,但是可能会影响性能。想了想,其实用sql server提供的计算列,就可以轻松实现这个需求。
例子如下:员工表,有字段:人员id,姓名,人员编码,人员入职时间,现在希望要增加一个字段显示工龄,就是在公司工作的时间,如 1.5年。
CREATE TABLE emp
(
emp_id INT PRIMARY KEY ,
emp_name NVARCHAR(10) not null,
emp_code VARCHAR(20) not null,
hire_date DATE not null
);
实现方法就是新增一个计算列:
alter table emp
add employment_time as cast(datediff(month,hire_date,GETDATE())*1.0/12 as numeric(8,1));
三、计算字段的应用
1、计算字段
计算字段存储在数据库表里面的数据一般会出现不是应用程序所需要的的数据格式,下面举3个简单的例子。
(1)如果想要在一个字段中既要显示公司的名,又要显示公司地址,但是这两个信息一般会包含在不同的列中。
(2)比如物品订单表存储物品的价格和数量,但不需要存储每个物品的总价格,有时候为了打印发票,所以需要物品的总价格。
(3)比如需要根据表数据进行总数,平均数和最大值和最小值或者其他的计算。
在上述的三个例子中,存储在表中的数据都不是应用程序所需要的,这个时候我们就需要直接从数据库中检索出转换然后计算或者格式化数据,而不是检索出数据,然后在客户机应用程序中重新格式化数据了。
这个时候就要发挥计算字段的作用了,计算字段不是存储在数据库表里面的,而是在运行SELECT语句内创建的,需要注意的是 MySQL 是可以分清哪些是数据库字段,哪些是计算字段的。
2、拼接字段
为了更好的说明如何使用计算字段,下面会顺便介绍拼接字段,然后连个一起举例展示。
(1)需求:vendors 表包含供应商名和位置信息。假如要生成一个供应商报表, 需要在供应商的名字中按照 name(location) 这样的格式列出供应商的位置。此报表需要单个值,而表中数据存储在两个列 vend_name 和 vend_ country 中。此外,需要用括号将 vend_country 括起来,这些东西都没有明确存储在数据库表中。我们来看看怎样编写返回供应商名和位置的 SELECT语句。
(2)解决办法是:把两个列拼接起来。在MySQL的SELECT语句中,可使用 Concat() 函数来拼接两个列。Concat()拼接串,即把多个串连接起来形成一个较长的串。看下面的sql语句和执行结果。
分析一下上面的 sql 语句的,首先 Concat() 需要一个或多个指定的串,各个串之间用逗号分隔。 上面的SELECT语句连接以下4个元素:分别是,存储在vend_name列中的名字;包含一个空格和一个左圆括号的串;存储在vend_country列中的国家;包含一个右圆括号的串。
从上述输出中可以看到,SELECT语句返回包含上述4个元素的单个列 (计算字段)。这个就是拼接字段和计算字段的使用方法。
3、执行算术计算
(1)需求:orders 表包含收到的所有订单,orderitems 表包含每个订单的各项物品,下面要查出订单号为20005的所有物品。
先查询订单号为20005的所有物品
item_price列包含订单中每项物品的单价。如下汇总物品的价格(单价乘以订购数量)
从输出的结果我们可以看出,expanded_price 列为一个计算字段展示,此列的计算值为 quantity*item_price,客户机可以直接使用这个计算出来的结果,而不是查询出数量和价格然后在客户机里面重新定义重新计算。
内容捯饬当然客户机是完全可以进行这些计算和格式化数据的,但是一般来说,在数据库服务器上面完成这些操作要比在客户机中完成要快得多,因为DBMS是设计快速有效地完成这种处理的。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 基于Microsoft.Extensions.AI核心库实现RAG应用
· Linux系列:如何用heaptrack跟踪.NET程序的非托管内存泄露
· 开发者必知的日志记录最佳实践
· SQL Server 2025 AI相关能力初探
· 震惊!C++程序真的从main开始吗?99%的程序员都答错了
· 单元测试从入门到精通
· 【硬核科普】Trae如何「偷看」你的代码?零基础破解AI编程运行原理
· 上周热点回顾(3.3-3.9)
· winform 绘制太阳,地球,月球 运作规律