обновление строк таблицы в postgres с использованием подзапроса

141

Используя postgres 8.4, моя цель - обновить существующую таблицу:

CREATE TABLE public.dummy
(
  address_id SERIAL,
  addr1 character(40),
  addr2 character(40),
  city character(25),
  state character(2),
  zip character(5),
  customer boolean,
  supplier boolean,
  partner boolean

)
WITH (
  OIDS=FALSE
);

Изначально я тестировал свой запрос с помощью инструкции insert:

insert into address customer,supplier,partner
SELECT  
    case when cust.addr1 is not null then TRUE else FALSE end customer, 
    case when suppl.addr1 is not null then TRUE else FALSE end supplier,
    case when partn.addr1 is not null then TRUE else FALSE end partner
from (
    SELECT *
        from address) pa
    left outer join cust_original cust
        on (pa.addr1=cust.addr1 and pa.addr2=cust.addr2 and pa.city=cust.city 
            and pa.state=cust.state and substring(cust.zip,1,5) = pa.zip  )
    left outer join supp_original suppl 
        on (pa.addr1=suppl.addr1 and pa.addr2=suppl.addr2 and pa.city=suppl.city 
                and pa.state=suppl.state and pa.zip = substring(suppl.zip,1,5))
    left outer join partner_original partn
        on (pa.addr1=partn.addr1 and pa.addr2=partn.addr2 and pa.city=partn.city
                  and pa.state=partn.state and pa.zip = substring(partn.zip,1,5) )
where pa.address_id = address_id

быть новичком. Я не могу преобразовать в инструкцию обновления, т.е. обновлять существующие строки со значениями, возвращаемыми выражением select. Любая помощь приветствуется.

  • 0
    У вас есть какой-либо идентификатор в адресной таблице, который может быть использован для определения того, что строка существует?
  • 0
    да, но его система генерируется.
Теги:
sql-update
subquery

3 ответа

344
Лучший ответ

Postgres позволяет:

UPDATE dummy
SET customer=subquery.customer,
    address=subquery.address,
    partn=subquery.partn
FROM (SELECT address_id, customer, address, partn
      FROM  /* big hairy SQL */ ...) AS subquery
WHERE dummy.address_id=subquery.address_id;

Этот синтаксис не является стандартным SQL, но гораздо удобнее для этого типа запросов, чем стандартный SQL. Я считаю, что Oracle (по крайней мере) принимает нечто подобное.

  • 0
    кажется, что я пытаюсь что-то другое, например. если есть 3 столбца bool c1, c2, c3, изначально все установлено в false. но на основе подзапроса устанавливаются в true. обновить набор c1 = TRUE, где id in (подзапрос1), установить c2 = TRUE, где id in (подзапрос2), установить c3 = True, где id in (подзапрос3). Я был успешным, когда я разделил это на 3 обновления, но я не уверен, как достичь результата с помощью одного обновления. надеюсь, что это имеет смысл.
  • 3
    Кстати, Oracle принимает эту базовую конструкцию, однако производительность обновления имеет тенденцию к значительному снижению по мере увеличения таблиц. Это нормально, хотя Oracle также поддерживает оператор MERGE.
Показать ещё 6 комментариев
62

Вы после синтаксиса UPDATE FROM.

UPDATE 
  table T1  
SET 
  column1 = t2.column1 
FROM 
  table t2 
  INNER JOIN table t3 USING (column2) 
WHERE 
  t1.column2 = t2.column2;

Ссылки

3

Если при использовании соединения нет прироста производительности, я предпочитаю Common Table Expressions (CTE) для чтения:

WITH subquery AS (
    SELECT address_id, customer, address, partn
    FROM  /* big hairy SQL */ ...
)
UPDATE dummy
SET customer=subquery.customer,
    address=subquery.address,
    partn=subquery.partn
FROM subquery
WHERE dummy.address_id=subquery.address_id;

ИМХО немного более современно.

  • 1
    Синтаксис не совместим со старыми версиями Postgres до v9.1 (см postgresql.org/docs/9.1/static/sql-update.html и предыдущие версии) я на v8.2, так что у вас есть поместить весь оператор CTE / With в скобки после ключевого слова FROM, и это сработает.

Ещё вопросы

Сообщество Overcoder
Наверх
Меню