MySQL SELECT同时UPDATE同一张表

MySQL不容许SELECT FROM后面指向用做UPDATE的表,有时候让人纠结。固然,有比建立无休止的临时表更好的办法。本文解释如何UPDATE一张表,同时在查询子句中使用SELECT.

问题描述

假设我要UPDATE的表跟查询子句是同一张表,这样作有许多种缘由,例如用统计数据更新表的字段(此时须要用group子句返回统计值),从某一条记录的字段update另外一条记录,而没必要使用非标准的语句,等等。举个例子:
[sql]  view plain copy
  1. create table apples(variety char(10) primary key, price int);  
  2.   
  3. insert into apples values('fuji', 5), ('gala', 6);  
  4.   
  5. update apples  
  6.     set price = (select price from apples where variety = 'gala')  
  7.     where variety = 'fuji';  

错误提示是:ERROR 1093 (HY000): You can't specify target table 'apples' for update in FROM clause. MySQL手册 UPDATE documentation 这下面有说明 : “Currently, you cannot update a table and select from the same table in a subquery.”
在这个例子中,要解决问题也十分简单,但有时候不得不经过查询子句来update目标。好在咱们有办法。

解决办法

既然MySQL是经过临时表来实现FROM子句里面的嵌套查询,那么把嵌套查询装进另一个嵌套查询里,可以使FROM子句查询和保存都是在临时表里进行,而后间接地在外围查询被引用。下面的语句是正确的:
[sql]  view plain copy
  1. update apples  
  2.    set price = (  
  3.       select price from (  
  4.          select * from apples  
  5.       ) as x  
  6.       where variety = 'gala')  
  7.    where variety = 'fuji';  

若是你想了解更多其中的机制,请阅读 MySQL Internals Manual 相关章节。

没有解决的问题

一个常见的问题是,IN()子句优化废品,被重写成相关的嵌套查询,有时(每每?)形成性能低下。把嵌套查询装进另一个嵌套查询里并不能阻止它重写成相关嵌套,除非我下狠招。这种状况下,最好用JOIN重构查询( rewrite such a query as a join )。

另外一个没解决的问题是临时表被引用屡次。“装进嵌套查询”的技巧没法解决这些问题,由于它们在编译时被建立,而上面讨论的update问题是在运行时。(译者注:我的认为跟文章讨论的主题没几毛钱关系)


原文地址 http://www.xaprb.com/blog/2006/06/23/how-to-select-from-an-update-target-in-mysql/
相关文章
相关标签/搜索