一.为什么要获取insert的id

写了测试类测试插入,插入之后用select查询出来进行Assert

插入成功后,不管Select对比的结果成功还是失败,都希望删除掉测试插入的结果

二.运行环境

mysql自增主键

mapper中的insert下是,这是通过mybatis_generator自动生成的,最新版本1.3.6.

   <selectKey keyProperty="id" order="BEFORE" resultType="java.lang.Long">
SELECT LAST_INSERT_ID()
</selectKey>

然后

 goodsClassService.insertGoodsClass(goodsclass);

但是用断点调试发现goodsclass的id一直是0,这让我不得其解.

三.原因

感觉Mybatis的这个自动生成的xml有问题.

参考:http://blog.csdn.net/slvher/article/details/42298355

6. last_insert_id()的值是由MySQL server来维护的,而且是为每条连接维护独立的值,也即,某条连接调用last_insert_id()获取到的值是这条连接最近一次insert操作执行后的自增值,该值不会被其它连接的sql语句所影响。这个行为保证了不同的连接能正确地获取到它最近一次insert sql执行所插入的行的自增值,也就是说,last_insert_id()的值不需要通过加锁或事务机制来保证其在多连接场景下的正确性。

mybatis的连接和客户端,比如workbench连接不同,last_insert_id()不能进行比较.

每一次测试运行都是新的连接,所last_insert_id()都是0,

mybatis默认生成的selectKey是在before之前,手动插入id为0,

如果给自增列插入0或者null,那么自增列设置为根据当前最大的id+1

如果批量插入就有问题了

Cause: com.mysql.jdbc.exceptions.jdbc4.MySQLIntegrityConstraintViolationException: Duplicate entry '66671' for key 'PRIMARY'

第一次插入id是0,可以自增.因为用了连接池,即使关闭了sqlsession也可能是同一个连接,第二次插入取出了last_insert_id()为66671,再插入,重复id错误.

三.解决办法

将上面的改为

 <selectKey keyProperty="id" order="AFTER" resultType="java.lang.Long">
SELECT LAST_INSERT_ID()
</selectKey>

这是插入之后取出SELECT LAST_INSERT_ID(),插入的时候id默认不插入,mysql会自动帮我们自增.

插入之后二中的goodsclass的id就变为last_id,可以取出来用了.

所以我觉得mybatis_generator这个配置操作有点小bug的

四.测试代码

省略了部分Assert代码

插入成功后,无论查询对比失败或者成功,都将插入的数据删掉.

//测试添加商品分类
//isok()返回状态码200
@Test
public void test_add_goodsClass() throws Exception
{
//测试插入是否成功
String ret=mockMvc.perform(post("/goods_class/add")
.param("pid","1"))
.andExpect(status().isOk())
.andReturn().getResponse().getContentAsString();
ObjectMapper objectMapper=new ObjectMapper();
Map<String,String> map=objectMapper.readValue(ret,Map.class);
Assert.assertNotNull(map);
Long last_insert_id=Long.valueOf(map.get("id"));
Assert.assertNotEquals(last_insert_id,Long.valueOf(0));
try {
 
//测试插入结果
Goodsclass goodsclass= goodsClassService.getGoodsClassById(last_insert_id);
Assert.assertEquals(goodsclass.getParent().getId(),Long.valueOf(1)); }
finally {
goodsClassService.deleteGoodsClassById(last_insert_id);
Assert.assertNull(goodsClassService.getGoodsClassById(last_insert_id));
} }

mybatis获取insert插入之后的id的更多相关文章

  1. myBatis获取批量插入数据的主键id

    在myBatis中获取刚刚插入的数据的主键id是比较容易的 , 一般来说下面的一句话就可以搞定了 , 网上也有很多相关资料去查. @Options(useGeneratedKeys = true, k ...

  2. mybatis获取刚刚插入到数据库的数据的id(转载)

    原文地址:https://blog.csdn.net/hehuihh/article/details/82800739 我用的是第一种写法,直接把代码copy到insert标签里(id要是自增id) ...

  3. 如何在dapper中获取刚插入行的ID

    二话不说: 1.先建立个表 CREATE TABLE [dbo].[UserInfo](    [ID] [int] IDENTITY(1,1) NOT NULL,    [UserName] [nc ...

  4. SQL Server output子句用法 output inserted.id 获取刚插入数据的id

    --插入数据,并返回刚刚插入的数据id INSERT INTO [soloreztest] ([name]) output inserted.id VALUES ('solorez') --执行结果: ...

  5. YII获取刚插入数据的id主键

    单条数据时model->attributes['id']; 循环插入时使用 Yii::app()->db->getLastInsertID() 获取 循环插入时需要每次插入后重置 m ...

  6. Mybatis中insert返回主键ID

    记录解决的过程,这里就不搬砖了. 1.获取insert后的主键id 原文链接:http://www.cnblogs.com/fsjohnhuang/p/4078659.html 2.insert后返回 ...

  7. MyBatis在insert插入操作时返回主键ID的配置

    在使用MyBatis做持久层时,insert语句默认是不返回记录的主键值,而是返回插入的记录条数:如果业务层需要得到记录的主键时,可以通过Mapper.XML配置的方式来完成这个功能. 在 INSER ...

  8. mybatis获取批量插入的主键自增id

    一.写一个实体类 public class UserInfo { private long userId; private String userAccount; private String use ...

  9. mybatis 获取insert返回的主键

    在我们开发过程中,在插入数据到数据库时,很多时候都需要把其主键返回,这里就说一下mybatis是怎么获取的. 其中mysql和oracle是不同的做法,因为mysql本身就提供字段自增的属性,而ora ...

随机推荐

  1. css基础教程

    css规则有两个主要部分构成:选择器,以及一条或多条声明. 值的不同写法和单位: 可以使用十六进制设置颜色值:#ff0000; 为节约字节,使用css缩写形式:#f00: 类选择器:以一个点号显示. ...

  2. flask 电子邮件进阶实践-用模板发送163邮件

    电子邮件进阶实践 下面来学习构建邮件的HTML正文,并使用模板组织内容. 一封电子邮件的正文可以是纯文本(text/plain),也可以是HTML格式的文本(text/html).处于全面的考虑,一封 ...

  3. php 防跨站表单提交

    一种最优方式防跨站表单提交,用户限时token 就是生成一个随机且变换频繁加密字符串(可逆和不可逆).放在表单中,等到表单提交后检查. 这个随机字符串如果和当前用户身份相关联的话,那么攻击者伪造请求会 ...

  4. Springboot 实现多环境配置

    多环境配置 我们在开发Spring Boot应用时,通常同一套程序会被应用和安装到几个不同的环境,比如:开发.测试.生产等.其中每个环境的数据库地址.服务器端口等等配置都会不同,如果在为不同环境打包时 ...

  5. 通过宝塔webhook,实现git自动拉取服务器代码

    1.宝塔安装webhook,添加一条记录,脚本内容为: #!/bin/bash echo "" #输出当前时间 date --date='0 days ago' "+%Y ...

  6. 安卓websocket 封装基于Java-websocket

    本文借鉴于https://blog.csdn.net/u013872857/article/details/79190643感谢大神的贡献 借鉴文章所用websocket 第三方是:nv-websoc ...

  7. 火狐开发----如何快速的安装火狐XPI文件

    第一步:火狐的自动安装扩展程序,https://addons.mozilla.org/zh-CN/firefox/addon/autoinstaller/ 第二步:安装wget工具,这个Linux应该 ...

  8. Leetcode 树(102, 637)

    637: 二叉树的层平均值 给定一个非空二叉树,返回一个由每层节点平均值组成的数组: https://leetcode-cn.com/problems/average-of-levels-in-bin ...

  9. SQL-递归查询在Ora与Mssql

    今天在工作中,有同事“请教”从 Sql Server 移植数据到 DM DB 的改写问题,本以为难度不大,结果发现 Sql Server 数据库的语法.架构上,与 Oracle / DM 数据库差异还 ...

  10. 你不知道的JS(2)深入了解闭包

    很久之前就想写一篇关于闭包的博客了,但是总是担心写的不够完全.不够好,不管怎样,还是要把我理解的闭包和大家分享下,比较长,希望耐心看完. 定义 说实话,给闭包下一个定义是很困难的,原因在于javasc ...