本站首页    管理页面    写新日志    退出


«August 2025»
12
3456789
10111213141516
17181920212223
24252627282930
31


公告

戒除浮躁,读好书,交益友


我的分类(专题)

日志更新

最新评论

留言板

链接

Blog信息
blog名称:邢红瑞的blog
日志总数:523
评论数量:1142
留言数量:0
访问次数:9701078
建立时间:2004年12月20日




[数据库]Mysql的LAST_INSERT_ID
原创空间,  文章收藏,  网上资源,  软件技术

邢红瑞 发表于 2007/6/4 13:52:02

 自动返回最后一个 INSERT 或 UPDATE 操作为 AUTO_INCREMENT 列设置的第一个发生的值.LAST_INSERT_ID是基于单个connection的, 不可能被其它的客户端连接改变。用 SELECT LAST_INSERT_ID(); 查询LAST_INSERT_ID的值.使用单INSERT语句插入多条记录,  LAST_INSERT_ID只返回插入的第一条记录产生的值. 比如mysql> INSERT INTO t VALUES (NULL, 'aaaa'), (NULL, 'bbbb'), (NULL, 'cccc');    mysql> SELECT * FROM t;    +----+------+    | id | name |    +----+------+    |  1 | Bob  |    |  2 | aaaa |    |  3 | bbbb |    |  4 | cccc |    +----+------+    mysql> SELECT LAST_INSERT_ID();    +------------------+    | LAST_INSERT_ID() |    +------------------+    |                2 |    +------------------+   ID 2 是在插入第一条记录aaaa 时产生的. LAST_INSERT_ID 是与table无关的,如果向表a插入数据后,再向表b插入数据,LAST_INSERT_ID会改变。 一般情况下获取刚插入的数据的id,使用select max(id) from table 是可以的。 但在多线程情况下,就不行了。在多用户交替插入数据的情况下max(id)显然不能用。使用LAST_INSERT_ID了,因为LAST_INSERT_ID是基于Connection的,只要每个线程都使用独立的Connection对象,LAST_INSERT_ID函数将返回该Connection对AUTO_INCREMENT列最新的insert or update操作生成的第一个record的ID。这个值不能被其它客户端(Connection)影响,保证了你能够找回自己的 ID 而不用担心其它客户端的活动,而且不需要加锁。LAST_INSERT_ID() LAST_INSERT_ID(expr) 自动返回最后一个INSERT或 UPDATE 问询为 AUTO_INCREMENT列设置的第一个 发生的值。 mysql> SELECT LAST_INSERT_ID();         -> 195 产生的ID 每次连接后保存在服务器中。这意味着函数向一个给定客户端返回的值是该客户端产生对影响AUTO_INCREMENT列的最新语句第一个 AUTO_INCREMENT值的。这个值不能被其它客户端影响,即使它们产生它们自己的 AUTO_INCREMENT值。这个行为保证了你能够找回自己的 ID 而不用担心其它客户端的活动,而且不需要加锁或处理。 使用一个非“magic”值来更新某一行的AUTO_INCREMENT 列,则LAST_INSERT_ID() 的值不会变化( 一个不是 NULL也不是 0的值)。使用单INSERT语句插入多个行,  LAST_INSERT_ID() 只返回插入的第一行产生的值。其原因是这使依靠其它服务器复制同样的 INSERT语句变得简单。 例如: mysql> USE test;Database changedmysql> CREATE TABLE t (    ->   id INT AUTO_INCREMENT NOT NULL PRIMARY KEY,    ->   name VARCHAR(10) NOT NULL    -> );Query OK, 0 rows affected (0.09 sec) mysql> INSERT INTO t VALUES (NULL, 'Bob');Query OK, 1 row affected (0.01 sec) mysql> SELECT * FROM t;+----+------+| id | name |+----+------+|  1 | Bob  |+----+------+1 row in set (0.01 sec) mysql> SELECT LAST_INSERT_ID();+------------------+| LAST_INSERT_ID() |+------------------+|                1 |+------------------+1 row in set (0.00 sec) mysql> INSERT INTO t VALUES    -> (NULL, 'Mary'), (NULL, 'Jane'), (NULL, 'Lisa');Query OK, 3 rows affected (0.00 sec)Records: 3  Duplicates: 0  Warnings: 0 mysql> SELECT * FROM t;+----+------+| id | name |+----+------+|  1 | Bob  ||  2 | Mary ||  3 | Jane ||  4 | Lisa |+----+------+4 rows in set (0.01 sec) mysql> SELECT LAST_INSERT_ID();+------------------+| LAST_INSERT_ID() |+------------------+|                2 |+------------------+1 row in set (0.00 sec)虽然第二个问询将3 个新行插入 t, 对这些行的第一行产生的 ID 为 2, 这也是 LAST_INSERT_ID()返回的值。使用 INSERT IGNORE而记录被忽略,则AUTO_INCREMENT 计数器不会增量,而 LAST_INSERT_ID() 返回0, 没有插入任何记录查看源码my_ulonglong STDCALL mysql_insert_id(MYSQL *mysql){  return mysql->;last_used_con->;insert_id;} MYSQL是个结构体,里面包括数据库链接和一些当前数据库链接的状态值, 其中在MYSQL结构体里面有insert_id,mysql_insert_id函数返回的就是结构体里面的找个值。 typedef struct st_mysql{  NET           net;                    /* Communication parameters */  gptr          connector_fd;           /* ConnectorFd for SSL */  char          *host,*user,*passwd,*unix_socket,*server_version,*host_info,*info;  char          *db;  struct charset_info_st *charset;  MYSQL_FIELD   *fields;   MEM_ROOT      field_alloc;  my_ulonglong affected_rows;  my_ulonglong insert_id;               /* id if insert on table with NEXTNR */  my_ulonglong extra_info;              /* Used by mysqlshow */  unsigned long thread_id;              /* Id for connection in server */  unsigned long packet_length;            unsigned int  port;  unsigned long client_flag,server_capabilities;  unsigned int  protocol_version;  unsigned int  field_count;  unsigned int  server_status;  unsigned int  server_language;  unsigned int  warning_count;  struct st_mysql_options options;  enum mysql_status status;  my_bool       free_me;                /* If free in mysql_close */  my_bool       reconnect;              /* set to 1 if automatic reconnect */    /* session-wide random string */  char          scramble[SCRAMBLE_LENGTH+1];  /*   Set if this is the original connection, not a master or a slave we have   added though mysql_rpl_probe() or mysql_set_master()/ mysql_add_slave()*/  my_bool rpl_pivot;  /*    Pointers to the master, and the next slave connections, points to    itself if lone connection.  */  struct st_mysql* master, *next_slave;   struct st_mysql* last_used_slave; /* needed for round-robin slave pick *//* needed for send/read/store/use result to work correctly with replication */  struct st_mysql* last_used_con;   LIST  *stmts;                     /* list of all statements */  const struct st_mysql_methods *methods;  void *thd;  /*    Points to boolean flag in MYSQL_RES  or MYSQL_STMT. We set this flag    from mysql_stmt_close if close had to cancel result set of this object.  */  my_bool *unbuffered_fetch_owner;} MYSQL; 把insert_id理解成max(id)是错误的,因为有并发存在。结构体里面有insert_id是针对当前MYSQL连接的,而每次mysql_query操作在mysql服务器上是一次“原子”操作。


阅读全文(5414) | 回复(0) | 编辑 | 精华
 



发表评论:
昵称:
密码:
主页:
标题:
验证码:  (不区分大小写,请仔细填写,输错需重写评论内容!)



站点首页 | 联系我们 | 博客注册 | 博客登陆

Sponsored By W3CHINA
W3CHINA Blog 0.8 Processed in 0.328 second(s), page refreshed 144764953 times.
《全国人大常委会关于维护互联网安全的决定》  《计算机信息网络国际联网安全保护管理办法》
苏ICP备05006046号