10

我只是在编写一个看起来或多或少像这样的复杂更新查询:

update table join
    (select y, min(x) as MinX 
     from table
     group by y) as t1
    using (y)
set x = x - MinX

这意味着该变量x是基于子查询更新的,该子查询也处理变量x——但是这不能x被运行的更新命令修改吗?这不是问题吗?我的意思是,在正常的编程中,你通常必须明确地处理这个问题,即将新值从旧值存储到其他地方,工作完成后,用新值替换旧值......但是SQL 数据库将如何做到这一点?

我对单一的观察或实验不感兴趣。我想有一个来自 docs 或 sql 标准的片段,它会说明在这种情况下定义的行为是什么。我正在使用 MySQL,但对其他 PostgresQL、Oracle 等也有效的答案,特别是对于一般的 SQL 标准,我们表示赞赏。谢谢!

4

4 回答 4

5

**已编辑**

从目标表中选择

13.2.9.8 开始。FROM 子句中的子查询

FROM 子句中的子查询可以返回标量、列、行或表。FROM 子句中的子查询不能是相关子查询,除非在 JOIN 操作的 ON 子句中使用。

所以,是的,您可以执行上述查询。

问题

这里确实有两个问题。有并发性,或者确保没有其他人从我们脚下更改数据。这是通过锁定处理的。处理新旧值的实际修改是通过派生表处理的。

锁定

在上述查询的情况下,使用 InnoDB,MySQL 首先执行 SELECT,并分别获取表中每一行的读取(共享)锁。如果您在 SELECT 语句中有 WHERE 子句,那么只有您选择的记录会被锁定,而范围也会导致任何间隙被锁定。

读锁可防止任何其他查询获取写锁,因此当记录被读锁定时,无法从其他地方更新记录。

然后,MySQL 分别在表中的每条记录上获取一个写(排他)锁。如果您的 UPDATE 语句中有 WHERE 子句,那么只有特定记录将被写锁定,同样,如果 WHERE 子句选择了一个范围,那么您将锁定一个范围。

任何从前一个 SELECT 获得读锁的记录都会自动升级为写锁。

写锁可防止其他查询获得读锁或写锁。

您可以使用Innotop通过在 Lock 模式下运行它来查看这一点,启动事务,执行查询(但不要提交),您将在 Innotop 中看到锁。此外,您可以在没有 Innotop 的情况下使用SHOW ENGINE INNODB STATUS.

死锁

如果同时运行两个实例,您的查询很容易出现死锁。如果查询 A 获得了读锁,然后查询 B 获得了读锁,则查询 A 必须等待查询 B 的读锁释放才能获得写锁。但是,查询 B 直到完成后才会释放读锁,除非它可以获取写锁,否则它不会完成。查询 A 和查询 B 陷入僵局,因此陷入僵局。

因此,您可能希望执行显式表锁,既可以避免大量记录锁(这会占用内存并影响性能),也可以避免死锁。

另一种方法是在您的内部 SELECT 上使用 SELECT ... FOR UPDATE。这从所有行上的写锁开始,而不是从读取和升级它们开始。

派生表

对于内部 SELECT,MySQL 创建一个派生的临时表。派生表是存在于由 MySQL 自动创建的临时表中的数据的实际非索引副本(与您显式创建并可以添加索引的临时表相反)。

由于 MySQL 使用派生表,因此这是您在问题中引用的临时旧值。换句话说,这里没有魔法。MySQL 做它就像你在其他任何地方做的一样,有一个临时值。

您可以通过对您的 UPDATE 语句(在 MySQL 5.6+ 中支持)执行 EXPLAIN 来查看派生表。

于 2012-04-10T16:47:29.007 回答
2

A proper RDBMS uses statement level read consistency, which ensures the statement sees (selects) the data as it was at the time the statement began. So the scenario you are afraid of, won't occur.

Regards,
Rob.

于 2012-04-13T14:43:06.403 回答
1

Oracle 在 11.2文档中有这个

为每个查询提供一致的结果集,保证数据的一致性,无需用户进行任何操作。隐式查询(例如 UPDATE 语句中的 WHERE 子句隐含的查询)保证了一组一致的结果。但是,隐式查询中的每个语句都不会看到 DML 语句本身所做的更改,而是看到更改之前存在的数据。

于 2012-04-13T13:29:48.643 回答
0

尽管已经注意到您不应该根据自己的数据对表进行更新,但您应该能够调整 MySQL 语法以允许通过

update Table1, 
       (select T2.y, MIN( T2.x ) as MinX from Table1 T2 group by T2.y ) PreQuery
  set Table1.x = Table1.x - PreQuery.MinX
  where Table1.y = PreQuery.y

我不知道语法是否使用 JOIN 与逗号列表版本走不同的路线,但通过完整的预查询,您必须首先应用其结果完成一次,并加入(通过 WHERE)实际执行更新。

于 2012-04-10T18:02:41.680 回答