MySQL 更新字段的值为当前最大值加1

摘要:在MySQL中,我们在执行update或者insert语句时,可以借助select语句更新一个字段,使其值更新为该字段的最大值加1。

在《MySQL中自增长序列(@i:=@i+1)的用处及用法》中,介绍了如何在select语句中生成递增序列,《MySQL 把查询结果更新或者插入到新表》介绍了把查询到的多条记录复制到另一张表中,均没有介绍如何更新数据库字段的值为当前最大值加指定步长。所以在本文中,楼兰胡杨将带着大家了解如何更新某个字段的值为当前最大值加指定步长1,步长也可以是其它满足诉求的值。

创建测试表

使用以下 MySQL 语句创建测试表test并写入三条测试数据:

mysql 复制代码
CREATE TABLE test (
  `id` bigint(20) UNSIGNED NOT NULL PRIMARY KEY AUTO_INCREMENT COMMENT '主键ID',
  incr_id INT NOT NULL COMMENT '递增序列',
  creator varchar(30) NOT NULL COMMENT '创建者' 
) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_bin COMMENT = '测试表';

INSERT INTO test(incr_id,creator) VALUES (FLOOR(1 + RAND() * 100),"张三");
INSERT INTO test(incr_id,creator) VALUES (FLOOR(1 + RAND() * 100),"张三丰");
INSERT INTO test(incr_id,creator) VALUES (FLOOR(1 + RAND() * 100),"楼兰胡杨");

表中incr_id用于演示如何更新它的值为其当前最大值加1。当然,令其如主键ID一样支持AUTO_INCREMENT时,可以实现自增长,实现方案非常简单,本文不再赘述。这里主要讲述如何通过update select和insert select实现字段值递增的策略。

update select实现字段递增赋值

首先使用聚合函数max来计算最大值,然后将其加1。以下SQL将返回test表中incr_id字段的最大值并且加1:

sql 复制代码
SELECT MAX(incr_id)+1 FROM test;

运行此命令后,将返回一个包含最大值加1的单独的列。接下来,楼兰胡杨将此用作update语句的源值,以下实现方案基于嵌套子查询完成:

mysql 复制代码
UPDATE test 
SET incr_id = (
    SELECT max_incr FROM (
        SELECT MAX(incr_id) + 1 AS max_incr FROM test
    ) AS tmp
)
WHERE id = 3;

在这个更新DML中,我们在子查询语句中使用聚合函数max计算字段最大值,然后对其最大值加1。与前面的例子一样,WHERE子句用于指定要更新的记录。这个方案因为耗内存、效率低而不适用于批量更新场景,但变量模拟自增方案适用于批量更新或复杂逻辑控制,实现脚本如下:

mysql 复制代码
-- 自定义变量
SET @max_incr = (SELECT MAX(incr_id) + 1 FROM test);
-- 使用变量更新字段
UPDATE test SET incr_id = @max_incr WHERE id = 3;

insert select实现字段递增赋值

在insert操作中实现字段递增赋值时,也是首先使用聚合函数max来计算最大值,然后将其加1。先介绍一下insert select语法糖:

mysql 复制代码
INSERT INTO target_table (column1, column2, ...)
SELECT value1, value2, ...
FROM source_table_a a JOIN source_table_b b ON a.id = b.a_id
WHERE condition_clause;
  • target_table:用于插入数据的目标表。
  • source_table_a 和 source_table_b:源表,从中选择数据用于插入目标表。可以从多个表中选择数据并插入到目标表中,可以一个表。
  • value1, value2, ...:被插入到目标表的值,既可以是从源表中选择的列,也可以是返回常量的表达式,更可以是一个常量。在编写SELECT语句的时候,可以使用MySQL支持的全部语法。

下面使用上述语法糖新增一条记录,并且令incr_id字段的新值为其最大值加1:

mysql 复制代码
INSERT INTO test ( `incr_id`, `creator`) 
SELECT IFNULL(max(incr_id), 0) + 1, '递增序列' FROM test;

IFNULL是专门处理 NULL 值的。若目标表的某些列不允许为 NULL,而假设 SELECT 查询返回 NULL 值,会导致数据插入操作失败。这种书写格式与如下常见VALUES写法区别很大:

myssql 复制代码
INSERT INTO test(incr_id) VALUES (FLOOR(1 + RAND() * 100),"普通insert操作");

小结

本篇文章的内容基本上就是这些,我们来复盘一下。在本文中,我们提供了一些使用MySQL select语句更新字段的示例,这些DML可以使用聚合函数或者嵌套子查询来计算要设置的新值。各位老铁无论选择哪种方式,都需要确保更新操作仅更新需要更新的记录,否则您可能会无意中、错误地❌更改了整张表的记录。楼兰胡杨还提到,如果您的表包含大量记录,使用SELECT子查询操作可能会比较慢,而借助自定义变量计算最大值的方案可能更轻快。

欢迎点赞阅读,一同学习交流;若有疑问,请在文章下方留下你的神评妙论!以促使博文💯高质量。

相关推荐
C182981825751 小时前
幻想读 通过多版本并发控制(MVCC)和间隙锁(Gap Lock)的组合也能防止幻读具体说下
mysql
鲁子狄3 小时前
[笔记] 动态 SQL 查询技术解析:构建灵活高效的企业级数据访问层
java·spring boot·笔记·sql·mysql·mybatis
Dubhehug4 小时前
8.数据库索引
数据库·mysql·索引·索引分类·索引优缺点
冬夜戏雪5 小时前
阿里云ubuntu安装mysql docker容器(拉,运行,测试完整版)
mysql·ubuntu·阿里云
♡喜欢做梦5 小时前
【MySQL】索引
数据库·mysql
都叫我大帅哥5 小时前
📉 MySQL索引罢工事件簿:揭秘失效原因与优化起义方案
java·mysql
什么半岛铁盒5 小时前
MySQL数据库的基础操作
linux·运维·服务器·mysql
fengye2071618 小时前
板凳-------Mysql cookbook学习 (十一--------10)
学习·mysql·adb
kfepiza15 小时前
Debian-10编译安装Mysql-5.7.44 笔记250706
linux·数据库·笔记·mysql·debian·bash