Вопрос по sql-update, syntax, mysql – Синтаксис UPDATE с ORDER BY, LIMIT и несколькими таблицами

6

Изучение SQL, извините, если это элементарно. Пытаясь выяснить работуUPDATE решение для следующего псевдоиш-кода:

UPDATE tableA 
SET tableA.col1 = '$var'
WHERE tableA.user_id = tableB.id
AND tableB.username = '$varName'
ORDER BY tableA.datetime DESC LIMIT 1

Выше больше похожеSELECT синтаксис, но я в основном пытаюсь обновить значение одного столбца вlatest row таблицы tableA, где имя пользователя, найденное в tableB.username (представленном $ varName), связано с его идентификатором в tableB.id, который существует как идентификатор в tableA.user_id.

Надеюсь, это имеет смысл. Я предполагаю, чтоJOIN необходимо, но подзапросы кажутся хлопотными дляUPDATE, я понимаюORDER BY а такжеLIMIT закрыты, когда в UPDATE задействованы несколько таблиц ... Но мне нужна функциональность. Это можно обойти?

Немного смущен, заранее спасибо.

Ваш Ответ

2   ответа
15

чтобы вставить ORDER BY и LIMIT в предложение FROM как часть объединения. Это позволяет вам сначала найти точную строку, которую нужно обновить (ta.id), а затем зафиксировать обновление.

UPDATE tableA AS target
    INNER JOIN (
      SELECT ta.id
      FROM tableA AS ta
        INNER JOIN tableB AS tb ON tb.id = ta.user_id
        WHERE tb.username = '$varName'
        ORDER BY ta.datetime DESC
        LIMIT 1) AS source ON source.id = target.id
    SET col1 = '$var';

Шляпная подсказка барону Шварцу, a.k.a. Xaprb, за отличный пост на эту точную тему: http://www.xaprb.com/blog/2006/08/10/how-to-use-order-by-and-limit-on-multi-table-updates-in-mysql/

+1 за ответ на свой вопрос
Note: Это не работает для временных таблиц - из руководства:You cannot refer to a TEMPORARY table more than once in the same query.
0

update work_to_do as target
   inner join (
      select w. client, work_unit
      from work_to_do as w
         inner join eligible_client as e on e.client = w.client
      where processor = 0
      order by priority desc
      limit 10
   ) as source on source.client = target.client
      and source.work_unit = target.work_unit
   set processor = @process_id;

Это работает отлично.

Похожие вопросы