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


304

Используя 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 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

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


У вас есть какой-либо идентификатор в адресной таблице, который может быть использован для определения того, что строка существует?
Андрей Адамович

да, но его система генерируется.
стек

Ответы:


683

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 (по крайней мере) принимает нечто подобное.


кажется, что я пытаюсь что-то другое, например. если есть 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.
gsiems

3
Это полностью не работает в postgresql 9.5, я получаюERROR: 42P01: relation "dummy" does not exist
user9645

73
dummyдолжен быть заменен именем таблицы, которую вы пытаетесь обновить. Пожалуйста, поймите вопрос и ответ, прежде чем пытаться подать заявку.
Андрей Лазарь

1
Возможно, стоит упомянуть, что в начале запроса нет необходимости указывать путь к столбцу с левой стороны, только в конце, иначе БД будет жаловаться с ошибкой: ссылка на столбец "address_id" неоднозначна
OJVM


51

Если при использовании объединения нет прироста производительности, я предпочитаю 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, и это сработает.
Spcogg второй

9

Есть много способов обновить строки.

Когда дело доходит до UPDATEстрок, использующих подзапросы, вы можете использовать любой из этих подходов.

  1. Подход-1 [Использование прямой ссылки на таблицу]
UPDATE
  <table1>
SET
  customer=<table2>.customer,
  address=<table2>.address,
  partn=<table2>.partn
FROM
  <table2>
WHERE
  <table1>.address_id=<table2>.address_i;

Объяснение: table1это таблица, которую мы хотим обновить, table2 это таблица, из которой мы получим значение, которое будет заменено / обновлено. Мы используем FROMпредложение, чтобы получить table2данные. WHERE Предложение поможет установить правильное отображение данных.

  1. Подход-2 [Использование подзапросов]
UPDATE
  <table1>
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;

Объяснение: Здесь мы используем подзапрос внутри FROMпредложения и присваиваем ему псевдоним. Так что он будет действовать как стол.

  1. Подход-3 [Использование нескольких объединенных таблиц]
UPDATE
  <table1>
SET
  customer=<table2>.customer,
  address=<table2>.address,
  partn=<table2>.partn
FROM
  <table2> as t2
  JOIN <table3> as t3
  ON
    t2.id = t3.id
WHERE
  <table1>.address_id=<table2>.address_i;

Объяснение: Иногда мы сталкиваемся с ситуацией, когда соединение таблиц так важно, чтобы получить правильные данные для обновления. Для этого Postgres позволяет нам объединять несколько таблиц внутри FROMпредложения.

  1. Подход-4 [Использование оператора WITH]

    • 4.1 [Используя простой запрос]
WITH subquery AS (
    SELECT
      address_id,
      customer,
      address,
      partn
    FROM
      <table1>;
)
UPDATE <table-X>
SET customer = subquery.customer,
    address  = subquery.address,
    partn    = subquery.partn
FROM subquery
WHERE <table-X>.address_id = subquery.address_id;
  • 4.2 [Использование запроса со сложным JOIN]
WITH subquery AS (
    SELECT address_id, customer, address, partn
    FROM
      <table1> as t1
    JOIN
      <table2> as t2
    ON
      t1.id = t2.id;
    -- You can build as COMPLEX as this query as per your need.
)
UPDATE <table-X>
SET customer = subquery.customer,
    address  = subquery.address,
    partn    = subquery.partn
FROM subquery
WHERE <table-X>.address_id = subquery.address_id;

Объяснение: В Postgres 9.1 эта WITHконцепция ( ) была введена. Используя это, мы можем делать любые сложные запросы и генерировать желаемый результат. Здесь мы используем этот подход для обновления таблицы.

Надеюсь, это будет полезно.



1

@Mayur "4.2 [Использование запроса со сложным JOIN]" с общими табличными выражениями (CTE) помог мне.

WITH cte AS (
SELECT e.id, e.postcode
FROM employees e
LEFT JOIN locations lc ON lc.postcode=cte.postcode
WHERE e.id=1
)
UPDATE employee_location SET lat=lc.lat, longitude=lc.longi
FROM cte
WHERE employee_location.id=cte.id;

Надеюсь, это поможет ...: D

Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.