关于Mysql LAST_INSERT_ID()使用
最近项目中出现了一个bug,(在插入多张关联的表的时候,出现小部分无法关联到需要的表格),查找了很久才发现这个问题的原因在此记录一下避免重复入坑。
原因就是因为mysql中LAST_INSERT_ID()的函数。
代码如下:
<selectKey resultType="java.lang.Integer"keyProperty="id" order="AFTER">
SELECT LAST_INSERT_ID()
selectKey>
insert into box_info (ID, MASTERBILL_ID, WAREHOUSE_PORT_ROUTING_ID,
TARE_WEIGHT, TOTAL_WEIGHT, WEIGHT_UNIT,
WAREHOUSE_LOCATION, LENGTH, WIDTH,
HEIGHT, LENGTH_UNIT, STATUS,
CREATE_DATE_TIME, CREATER, UPDATE_TIME,
UPDATER)
values (#{id,jdbcType=INTEGER}, #{masterbillId,jdbcType=INTEGER},#{warehousePortRoutingId,jdbcType=INTEGER},
#{tareWeight,jdbcType=DECIMAL}, #{totalWeight,jdbcType=DECIMAL},#{weightUnit,jdbcType=INTEGER},
#{warehouseLocation,jdbcType=INTEGER}, #{length,jdbcType=DECIMAL},#{width,jdbcType=DECIMAL},
#{height,jdbcType=DECIMAL}, #{lengthUnit,jdbcType=INTEGER},#{status,jdbcType=INTEGER},
#{createDateTime,jdbcType=TIMESTAMP}, #{creater,jdbcType=INTEGER},#{updateTime,jdbcType=TIMESTAMP},
#{updater,jdbcType=INTEGER})
因为业务需求box_info 主键id auto-increment 去除了,然后根据另一张表的id 赋值给box_info,后边通过SELECT LAST_INSERT_ID()获取id作为主单表关联,问题来了,由于去除了主键id 的auto-increment属性,导致获取的id为上个带有主键带有 auto-increment 属性的表格 insert或者 update的id,所以在关联查询的时候无法查询到相应的箱子信息
一个简单的例子:
代码如下:
$query="INSERT INTO `testtable` (`clou1`,`clou2`) VALUES('testvalue','test')";
mysql_query($query);
$query="SELECT LAST_INSERT_ID()";
$result=mysql_query($query);
$rows=mysql_fetch_row($result);
echo $rows[0];
这个函数是基于connection的,也就是不会被其他客户端的connection影响到,所以结果是准确的。如果使用select max(id)from table,在高密度的插入请求下,是有可能出问题的,返回错误值
LAST_INSERT_ID说明
从名字可以看出,LAST_INSERT_ID即为最后插入的ID值,根据MySQL的官方手册说明,它有2种使用方法
一是不带参数:LAST_INSERT_ID(),这种方法和AUTO_INCREMENT属性一起使用,当往带有‘AUTO_INCREMENT'属性字段的表有数据更新时(insert,update),LAST_INSERT_ID()即返回该字段的值;
二是带有表达式:如上面介绍的LAST_INSERT_ID(value+1),它返回的是表达式的值,即‘value+1';
LAST_INSERT_ID() 自动返回最后一个INSERT或 UPDATE 查询中 AUTO_INCREMENT列设置的第一个表发生的值。
MySQL的LAST_INSERT_ID的注意事项:
第一、查询和插入所使用的Connection对象必须是同一个才可以,否则返回值是不可预料的。
mysql> SELECTLAST_INSERT_ID();
-> 100
使用这函数向一个给定Connection对象返回的值是该Connection对象产生对影响AUTO_INCREMENT列的最新语句第一个AUTO_INCREMENT值的。这个值不能被其它Connection对象的影响,即它们产生它们自己的AUTO_INCREMENT值。
第二、LAST_INSERT_ID是与table无关的,如果向表a插入数据后,再向表b插入数据,LAST_INSERT_ID返回表b中的Id值。
第三、 假如你使用一条INSERT语句插入多个行, LAST_INSERT_ID() 只返回插入的第一行数据时产生的值。其原因是这使依靠其它服务器复制同样的 INSERT语句变得简单。
mysql> INSERTINTO t VALUES
-> (NULL, ‘Mary'),(NULL, ‘Jane'), (NULL, ‘Lisa');
mysql> SELECT *FROM t;
| id | name |
+—-+——+
| 1 |Bob |
| 2 | Mary |
| 3 | Jane |
| 4 | Lisa |
mysql> SELECTLAST_INSERT_ID(); //这就是我要说明的关键问题。
| LAST_INSERT_ID()|
| 2 |
虽然将3 个新行插入 t, 对这些行的第一行产生的 ID 为2, 这也是 LAST_INSERT_ID()返回的值。