关于Mysql LAST_INSERT_ID()使用

来源:互联网 发布:免费汽车维修软件 编辑:程序博客网 时间:2024/06/01 23:00

关于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()返回的值。