mysql - sqlfiddle 中的 Bug 或者我不明白 INSERT ... ON DUPLICATE KEY UPDATE

标签 mysql sql sqlfiddle

这个 sqlfiddle 链接似乎在调用之间保留状态: http://sqlfiddle.com/#!2/125bc/1

它包含 mysql 的这个模式:

CREATE TABLE if not exists  `standings` (
  `teamid` int(11) NOT NULL,
  `win` decimal(23,0) default NULL,
  `tie` decimal(23,0) default NULL,
  `lose` decimal(23,0) default NULL,
  PRIMARY KEY  (`teamid`)
) ENGINE=MyISAM DEFAULT CHARSET=utf8;


INSERT INTO `standings` (`teamid`, `win`, `tie`, `lose`) VALUES
(1, 3, 0, 0),
(2, 1, 1, 1),
(3, 1, 1, 1),
(4, 0, 0, 3);


CREATE TABLE if not exists `newscores` (
  `recordid` int(11) NOT NULL auto_increment,
  `teamid` int(11) NOT NULL,
  `gameid` int(11) NOT NULL,
  `totalstrokes` int(11) NOT NULL,
  primary key (recordid)
) ENGINE=MyISAM  DEFAULT CHARSET=utf8 AUTO_INCREMENT=5 ;



INSERT INTO `newscores` (`recordid`, `teamid`, `gameid`, `totalstrokes`) VALUES
(5, 1, 1, 145),
(6, 2, 1, 105),
(7, 5, 1, 145),
(8, 6, 1, 155);

这两个查询:

Insert into  standings (teamid, win, tie, lose) (SELECT g1.teamid, SUM( IF( g1.totalstrokes < g2.totalstrokes, 1, 0 ) ) AS win, SUM( IF( g1.totalstrokes = g2.totalstrokes, 1, 0 ) ) AS tie, SUM( IF( g1.totalstrokes > g2.totalstrokes, 1, 0 ) ) AS lose
FROM newscores g1
JOIN newscores g2 ON g1.gameid = g2.gameid
AND g1.teamid != g2.teamid
GROUP BY g1.teamid) 
on duplicate key 
update win = win + values(win), 
lose = lose + values(lose), 
tie = tie + values(tie);

select * from standings;

它应该产生这个 standings 表:

teamid   win tie lose
  1       4   1   1
  2       4   1   1
  3       1   1   1
  4       0   0   3
  5       1   1   1
  6       0   0   3

确实如此,第一次运行时。第二次运行时(通过将 url 粘贴到新窗口中),teamids 1、2、5、6 的一些值会增加。

我可以通过编辑架构将“创建表”更改为“如果表不存在则创建”,反之亦然,或者包括“删除表”语句来重置它。但这看起来很奇怪。

PS 如果有人比我更有意义地向这个问题添加 sqlfiddle 标签,我将不胜感激。

最佳答案

我是 SQL Fiddle 的作者。对此行为有一个非常简单的解释。本质上,您在右侧面板(“运行查询”端)上运行的所有查询都在一个事务 block 中执行,该事务 block 会立即回滚以保留模式的状态(如左侧面板所定义的)控制板)。大多数时候,一切都准确地返回到事情开始的地方,因为大多数在事务中执行的一切都可以回滚。然而,正如我早期使用 SQL Fiddle 发现的那样,少数不会回滚的事情之一是自动递增 ID 值。这是所有供应商有意做出的设计决定,旨在确保您的主键永远不会因重用而发生冲突。

由于自动 ID 应该是无意义的,所以这件事发生变化从来没有真正让我觉得是 SQL Fiddle 的问题(特别是因为大多数人无论如何都在左侧运行他们的插入)。

您无需打开多个窗口即可看到此行为,顺便说一句 - 只需多次点击“运行查询”,您就会看到它发生了变化。您还可以在我的 SQL Server 示例 fiddle 中看到它:http://sqlfiddle.com/#!3/1fa93/1只需多次运行该查询,您就会看到 id 每次都发生变化。 (如果一段时间未使用 fiddle ,它将重置,因为在特定 fiddle 未使用后我会删除正在运行的数据库以释放内存)。

编辑

呸!抱歉,没有仔细阅读您的查询。我现在明白是怎么回事了。您正在使用 MyISAM 表而不是 InnoDB。我真的需要禁用它们作为 MySQL 的一个选项。你看,他们根本不支持交易。因此,当我尝试回滚您对这些表的更新时,更新根本不会回滚。你不断增加它们。当我第一次推出 MySQL 作为一个选项时,我看到了这一点,因为默认情况下 MySQL 使用 MyISAM。我必须将默认值更改为 InnoDB 才能使事务正常工作。

在这里,当我简单地删除您对 MyISAM 的显式使用时,您可以看到它工作正常:http://sqlfiddle.com/#!2/cec6e/1

更新

由于对 MySQL 的一些更改要求,这里是我稍微重新安排的版本:http://sqlfiddle.com/#!2/a9cc3/1

关于mysql - sqlfiddle 中的 Bug 或者我不明白 INSERT ... ON DUPLICATE KEY UPDATE,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10606325/

相关文章:

sql - 计算销售价格以实现至少 10% 的利润

Mysql 条件求和

Mysql,数据透视表在 sqlfiddle 中工作但在 phpmyadmin 中不工作

mysql - FeathersJS 似乎将查询限制为 1000,即使您将 $limit 设置为高于 1000

mysql - 我可以对包含各种长度字母数字值的 VARCHAR 字段实现自然排序吗?

mysql - 根据指定列选择唯一行

mysql - 无法在 mySQL 中构建架构

mysql - 在 SqlFiddle 上执行触发器存储过程。 mysql

php - LIKE 的 SQL 错误

php - 以unicode在数据库中输入值