mysql实现不存在就插入,存在就更新,sql直接执行和mybatis实现的坑!

本文涉及的产品
云数据库 RDS MySQL,集群系列 2核4GB
推荐场景:
搭建个人博客
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
云数据库 RDS MySQL,高可用系列 2核4GB
简介: insert into ... on duplicate key update 字段=新值, mybatis执行报错: SQLException: No value specified for parameter 4,你甚至惊奇的发现你只传了3个参数却提示没找到第4个参数......亲身经历什么叫一个bug找一天

需求背景:数据表中有物理主键id,按照每次会话保存笔记,这里session_id作为每次会话的凭证,所以每次会话中可能会不断更新笔记,笔记存在就更新笔记,不存在就插入笔记

我想大家都会用

insert into 表名(字段1,字段2,...) values(值1, 值2,...) on duplicate key update 字段1=值1, 字段2=值2...

上面语法在mysql直接执行sql语句是没问题的,但是mybatis就有大坑。我个人完全不推荐这么用等号赋值,而是用values,在文章末尾会给出推荐写法。

该语句是基于唯一索引或主键使用,比如一个字段session_id被加上了unique index,并且表中已经存在了该session_id的记录值,那么插入就会更新。如果是物理主键id,那就参数需要带上这个id,不然id递增就会成为新记录

INSERT INTO my_table (user_id, kyc_info, todo_info) VALUES ("u123","客户信息","笔记")
ON DUPLICATE KEY UPDATE kyc_info= "客户信息", todo_info="笔记";

当插入session_id这个唯一索引重复的记录的时候,更新kyc_infotodo_info,如果是新记录,就直接插入。

其实这就相当于

-- 如果session_id相同代表是同一次会话,需求是笔记以会话为单位,一次会话不管怎么保存只能有一个笔记。
UPDATE 表名 SET kyc_info="客户信息",  todo_info="笔记" WHERE session_id="huihua123";

直接运行上面这个sql一点问题都没有,那简直看起来正确极了,但是用到mybatis,就一直报错

    <insert id="insertOrUpdateByPrimaryKeySelective" parameterType="你的entity实体对象">
        insert into 表名
        <trim prefix="(" suffix=")" suffixOverrides=",">
          <if test="userId!= null">
            user_id,
          </if>
          <if test="kycInfo!= null">
            kyc_info,
          </if>
          <if test="todoInfo!= null">
            todo_info,
          </if>
        </trim>
        <trim prefix="values (" suffix=")" suffixOverrides=",">
          <if test="userId!= null">
            #{userId,jdbcType=VARCHAR},
          </if>
          <if test="kycInfo!= null">
            #{kycInfo,jdbcType=VARCHAR},
          </if>
          <if test="todoInfo!= null">
            #{todoInfo,jdbcType=VARCHAR},
          </if>
        </trim>
        ON DUPLICATE KEY UPDATE
        kyc_info= #{kycInfo,jdbcType=VARCHAR},
        todo_info=#{todoInfo,jdbcType=VARCHAR}
    </insert>

结果一直报错:

SQLException: No value specified for parameter 4

你甚至惊奇的发现你只传了3个参数却提示没找到第4个参数。

问题出在这里

        ON DUPLICATE KEY UPDATE
        kyc_info= #{kycInfo,jdbcType=VARCHAR},
        todo_info=#{todoInfo,jdbcType=VARCHAR}

你应该写成

        ON DUPLICATE KEY UPDATE
        kyc_info= values(kyc_info),
        todo_info=values(todo_info)

结果就正确了。

综上,sql语句就应该写成
insert into 表名(字段1,字段2,...) values(值1, 值2,...) on duplicate key update 字段1=values(字段1), 字段2=values(字段2)
而不是
insert into 表名(字段1,字段2,...) values(值1, 值2,...) on duplicate key update 字段1=值1, 字段2=值2...

后者只有sql单独运行可以,mybatis运行报错。

前者不管单独运行还是mybatis执行都是ok

所以下面就不推荐这么写

INSERT INTO my_table(user_id, kyc_info, todo_info) VALUES ("u123","客户信息","笔记")
ON DUPLICATE KEY UPDATE kyc_info= "客户信息", todo_info="笔记";

推荐写法如下:

INSERT INTO my_table(user_id, kyc_info, todo_info) VALUES ("u123","客户信息","笔记")
ON DUPLICATE KEY UPDATE kyc_info= values(kyc_info), todo_info=values(todo_info);

亲身经历什么叫一个bug找一天。老铁们点点关注,给你们踩坑了!



欢迎一键三连~



有问题请留言,大家一起探讨学习



----------------------Talk is cheap, show me the code-----------------------

相关实践学习
如何在云端创建MySQL数据库
开始实验后,系统会自动创建一台自建MySQL的 源数据库 ECS 实例和一台 目标数据库 RDS。
全面了解阿里云能为你做什么
阿里云在全球各地部署高效节能的绿色数据中心,利用清洁计算为万物互联的新世界提供源源不断的能源动力,目前开服的区域包括中国(华北、华东、华南、香港)、新加坡、美国(美东、美西)、欧洲、中东、澳大利亚、日本。目前阿里云的产品涵盖弹性计算、数据库、存储与CDN、分析与搜索、云通信、网络、管理与监控、应用服务、互联网中间件、移动服务、视频服务等。通过本课程,来了解阿里云能够为你的业务带来哪些帮助 &nbsp; &nbsp; 相关的阿里云产品:云服务器ECS 云服务器 ECS(Elastic Compute Service)是一种弹性可伸缩的计算服务,助您降低 IT 成本,提升运维效率,使您更专注于核心业务创新。产品详情: https://www.aliyun.com/product/ecs
目录
相关文章
|
10天前
|
SQL Java 数据库连接
mybatis使用四:dao接口参数与mapper 接口中SQL的对应和对应方式的总结,MyBatis的parameterType传入参数类型
这篇文章是关于MyBatis中DAO接口参数与Mapper接口中SQL的对应关系,以及如何使用parameterType传入参数类型的详细总结。
22 10
|
4天前
|
SQL 运维 关系型数据库
MySQL 运维 SQL 备忘
MySQL 运维 SQL 备忘录
20 1
|
9天前
|
Java 关系型数据库 MySQL
springboot学习五:springboot整合Mybatis 连接 mysql数据库
这篇文章是关于如何使用Spring Boot整合MyBatis来连接MySQL数据库,并进行基本的增删改查操作的教程。
14 0
springboot学习五:springboot整合Mybatis 连接 mysql数据库
|
14天前
|
SQL 存储 关系型数据库
SQL文件导入MySQL数据库的详细指南
数据库中的数据转移是一项常规任务,无论是在数据迁移过程中,还是在数据备份、还原场景中,导入导出SQL文件显得尤为重要。特别是在使用MySQL数据库时,如何将SQL文件导入数据库是一项基本技能。本文将详细介绍如何将SQL文件导入MySQL数据库,并提供一个清晰、完整的步骤指南。这篇文章的内容字数大约在
31 1
|
6天前
|
SQL 存储 关系型数据库
mysql 数据库空间统计sql
mysql 数据库空间统计sql
19 0
|
6天前
|
SQL 存储 关系型数据库
mysql SQL必知语法
本文详细介绍了MySQLSQL的基本语法,包括SELECT、FROM、WHERE、GROUPBY、HAVING、ORDERBY等关键字的使用,以及数据库操作如创建、删除表,数据类型,插入、查询、过滤、排序、连接和汇总数据的方法。通过学习这些内容,读者将能更好地管理和操
8 0
|
14天前
|
SQL 分布式计算 关系型数据库
Hadoop-24 Sqoop迁移 MySQL到Hive 与 Hive到MySQL SQL生成数据 HDFS集群 Sqoop import jdbc ETL MapReduce
Hadoop-24 Sqoop迁移 MySQL到Hive 与 Hive到MySQL SQL生成数据 HDFS集群 Sqoop import jdbc ETL MapReduce
54 0
|
10天前
|
Java 数据库连接 Maven
mybatis使用一:springboot整合mybatis、mybatis generator,使用逆向工程生成java代码。
这篇文章介绍了如何在Spring Boot项目中整合MyBatis和MyBatis Generator,使用逆向工程来自动生成Java代码,包括实体类、Mapper文件和Example文件,以提高开发效率。
36 2
mybatis使用一:springboot整合mybatis、mybatis generator,使用逆向工程生成java代码。
|
10天前
|
SQL JSON Java
mybatis使用三:springboot整合mybatis,使用PageHelper 进行分页操作,并整合swagger2。使用正规的开发模式:定义统一的数据返回格式和请求模块
这篇文章介绍了如何在Spring Boot项目中整合MyBatis和PageHelper进行分页操作,并且集成Swagger2来生成API文档,同时定义了统一的数据返回格式和请求模块。
22 1
mybatis使用三:springboot整合mybatis,使用PageHelper 进行分页操作,并整合swagger2。使用正规的开发模式:定义统一的数据返回格式和请求模块
|
17天前
|
前端开发 Java Apache
Springboot整合shiro,带你学会shiro,入门级别教程,由浅入深,完整代码案例,各位项目想加这个模块的人也可以看这个,又或者不会mybatis-plus的也可以看这个
本文详细讲解了如何整合Apache Shiro与Spring Boot项目,包括数据库准备、项目配置、实体类、Mapper、Service、Controller的创建和配置,以及Shiro的配置和使用。
136 1
Springboot整合shiro,带你学会shiro,入门级别教程,由浅入深,完整代码案例,各位项目想加这个模块的人也可以看这个,又或者不会mybatis-plus的也可以看这个