BBS.ChinaUnix.net
首页 | 新闻 | Linux | FreeBSD | AIX | Windows | 博客 | 论坛 | 存储 | 网络 | 人才 | Wiki | 资料 | 读书 | 手册 | 下载 | 空间 | 搜索
  免费注册 | 忘记密码 | 会员登录 | 搜索 | 帮助 



last_insert_id()不可靠

首页 » 论坛 » MySQL »  
[打印] [订阅] [收藏] [本帖文本页] [推荐此主题给朋友,立即获积分]
linuxkumao   帅哥
光明使者




CU编号: 513413
注册:2007-1-2
最后登录: 2009-06-05
帖子:970
精华:0

可用积分:990 (丰衣足食)
信誉积分:100
专家积分:0 (本版:0)
空间积分:0
推广积分:0

来自:上海
状态:...离线...

[个人空间] [短信] [博客]


1楼 发表于 2008-3-24 11:13 

情况如下:

mysql> create table t (c1 integer auto_increment primary key);
Query OK, 0 rows affected (0.02 sec)

mysql> select last_insert_id();
+------------------+
| last_insert_id() |
+------------------+
|                0 |
+------------------+
1 row in set (0.00 sec)

mysql> insert t value();
Query OK, 1 row affected (0.01 sec)

mysql> select * from t;
+----+
| c1 |
+----+
|  1 |
+----+
1 row in set (0.00 sec)

mysql> select last_insert_id();
+------------------+
| last_insert_id() |
+------------------+
|                1 |
+------------------+
1 row in set (0.00 sec)

mysql> insert t value(),(),();
Query OK, 3 rows affected (0.00 sec)
Records: 3  Duplicates: 0  Warnings: 0

mysql> select last_insert_id();   <-----   这里错了
+------------------+
| last_insert_id() |
+------------------+
|                2 |
+------------------+
1 row in set (0.00 sec)

mysql> select * from t;
+----+
| c1 |
+----+
|  1 |
|  2 |
|  3 |
|  4 |
+----+
4 rows in set (0.00 sec)

mysql> select last_insert_id();
+------------------+
| last_insert_id() |
+------------------+
|                2 |
+------------------+
1 row in set (0.00 sec)

mysql> insert t value();
Query OK, 1 row affected (0.02 sec)

mysql> select * from t;
+----+
| c1 |
+----+
|  1 |
|  2 |
|  3 |
|  4 |
|  5 |
+----+
5 rows in set (0.00 sec)

mysql> select last_insert_id();
+------------------+
| last_insert_id() |
+------------------+
|                5 |
+------------------+
1 row in set (0.00 sec)

mysql>

有无办法解决?是不是要设什么参数?是Bug吗?

谢谢!

[ 本帖最后由 linuxkumao 于 2008-3-24 11:19 编辑 ]



您对本贴的看法:鲜花[0] 臭蛋[0]

__________________________________

为还能有一席之地而努力!!


北京野狼   帅哥
法师



CU编号: 74822
注册:2003-7-24
最后登录: 2009-07-04
帖子:6628
精华:4

可用积分:6440 (富足长乐)
信誉积分:100
专家积分:11 (本版:0)
空间积分:0
推广积分:0

状态:...离线...

[个人空间] [短信] [博客]


2楼 发表于 2008-3-24 13:27 

没错啊。 last_insert_id不是auto_increment的最大值, 仅仅对当前链接有效



您对本贴的看法:鲜花[0] 臭蛋[0]
linuxkumao   帅哥
光明使者




CU编号: 513413
注册:2007-1-2
最后登录: 2009-06-05
帖子:970
精华:0

可用积分:990 (丰衣足食)
信誉积分:100
专家积分:0 (本版:0)
空间积分:0
推广积分:0

来自:上海
状态:...离线...

[个人空间] [短信] [博客]


3楼 发表于 2008-3-24 14:37 

谢谢!

我并没有断开过连接啊,是在同一个Session里的

last_insert_id不是auto_increment的最大值吗?那里怎么做才能得到auto_increment的最大值呢?



您对本贴的看法:鲜花[0] 臭蛋[0]

__________________________________

为还能有一席之地而努力!!


版主 yueliangdao0608   帅哥 ((上帝他爷))
广告杀手-法王
上帝他爷


CU奥运火炬传递手2008
CU编号: 242895
注册:2005-3-22
最后登录: 2009-07-03
帖子:19649
精华:10

可用积分:11367 (大富大贵)
信誉积分:135
专家积分:105 (本版:93)
空间积分:2
推广积分:0

来自:上帝他女人诞生的地方
状态:...保密...

[个人空间] [短信] [博客]


4楼 发表于 2008-3-24 21:00 

last_insert_id()
只的是上一次插入的自增ID。
如果是新建的表,它为0,
如果是刚插入的记录,那它就是max(id)


不是不可靠,而是你没有理解!



您对本贴的看法:鲜花[0] 臭蛋[0]

__________________________________


自从上帝被耶稣弄死之后,他老婆就是我的人了
----------------------------------

linuxkumao   帅哥
光明使者




CU编号: 513413
注册:2007-1-2
最后登录: 2009-06-05
帖子:970
精华:0

可用积分:990 (丰衣足食)
信誉积分:100
专家积分:0 (本版:0)
空间积分:0
推广积分:0

来自:上海
状态:...离线...

[个人空间] [短信] [博客]


5楼 发表于 2008-3-25 07:40 

谢谢!



您对本贴的看法:鲜花[0] 臭蛋[0]

__________________________________

为还能有一席之地而努力!!


北京野狼   帅哥
法师



CU编号: 74822
注册:2003-7-24
最后登录: 2009-07-04
帖子:6628
精华:4

可用积分:6440 (富足长乐)
信誉积分:100
专家积分:11 (本版:0)
空间积分:0
推广积分:0

状态:...离线...

[个人空间] [短信] [博客]


6楼 发表于 2008-3-25 10:37 

看了半天mysql的代码。mysql_insert_id是这么定义的
my_ulonglong STDCALL mysql_insert_id(MYSQL *mysql)
{
  return mysql->;last_used_con->;insert_id;
}

MYSQL是个结构体,里面包括数据库链接和一些当前数据库链接的状态值,

其中在MYSQL结构体里面有insert_id,mysql_insert_id函数返回的就是结构体里面的找个值。

QUOTE:
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服务器上可以理解为一次“原子”操作。

[ 本帖最后由 北京野狼 于 2008-3-25 10:59 编辑 ]



您对本贴的看法:鲜花[1] 臭蛋[0]
版主 yueliangdao0608   帅哥 ((上帝他爷))
广告杀手-法王
上帝他爷


CU奥运火炬传递手2008
CU编号: 242895
注册:2005-3-22
最后登录: 2009-07-03
帖子:19649
精华:10

可用积分:11367 (大富大贵)
信誉积分:135
专家积分:105 (本版:93)
空间积分:2
推广积分:0

来自:上帝他女人诞生的地方
状态:...保密...

[个人空间] [短信] [博客]


7楼 发表于 2008-3-25 10:45 



QUOTE:
原帖由 北京野狼 于 2008-3-25 10:37 发表
看了半天mysql的代码。mysql_insert_id是这么定义的
my_ulonglong STDCALL mysql_insert_id(MYSQL *mysql)
{
  return mysql->;last_used_con->;insert_id;
}

MYSQL是个结构体,里面包括数据库链接和一些 ...

对哈,忘了说在一个连接中。
野狼兄分析的对。



您对本贴的看法:鲜花[0] 臭蛋[0]

__________________________________


自从上帝被耶稣弄死之后,他老婆就是我的人了
----------------------------------

sunnyfun
天使




CU编号: 550758
注册:2007-4-12
最后登录: 2009-07-02
帖子:1020
精华:1

可用积分:2330 (小富即安)
信誉积分:100
专家积分:428 (本版:283)
空间积分:804
推广积分:0

状态:...离线...

[个人空间] [短信] [博客]


8楼 发表于 2008-3-25 13:13 



QUOTE:
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 changed
mysql> 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, 这反映出没有插入任何记录。

看上去楼主本来就是在同一个连接。原因应该是我标红字的地方。

结合ROW_COUNT() 很容易达到楼主需要的效果的。



您对本贴的看法:鲜花[0] 臭蛋[0]
156950928   帅哥
新手




CU编号: 259150
注册:2005-4-24
最后登录: 2009-03-29
帖子:3
精华:0

可用积分:12 (白手起家)
信誉积分:100
专家积分:0 (本版:0)
空间积分:0
推广积分:0

状态:...离线...

[个人空间] [短信] [博客]


9楼 发表于 2008-3-26 22:20 

现在真正理解 last_insert_id() 的含义了。



您对本贴的看法:鲜花[0] 臭蛋[0]

首页 » 论坛 » MySQL »


 


Copyright © 2001-2009 ChinaUnix.net All Rights Reserved     联系我们:

感谢所有关心和支持过ChinaUnix的朋友们    转载本站内容请注明原作者名及出处

京ICP证041476号


清除 Cookies - ChinaUnix - Archiver - WAP - TOP

Processed in 0.069142 second(s), 4 queries , Gzip enabled