Обновить несколько строк в одном запросе, используя PostgreSQL

192

Я ищу, чтобы обновить несколько строк в PostgreSQL в одном выражении. Есть ли способ сделать что-то вроде следующего?

UPDATE table 
SET 
 column_a = 1 where column_b = '123',
 column_a = 2 where column_b = '345'
newUserNameHere
источник
Я продолжаю пытаться найти это на той странице, но я не могу получить это. Я вижу, где вы можете обновить несколько строк, используя оператор where, но я не понимаю, как обновить несколько строк, каждая с собственным оператором where. Я также искал в Google и не нашел четкого ответа, так что я надеялся, что кто-нибудь сможет дать наглядный пример.
newUserNameHere
Извините моя ошибка. Обновлено.
ноль 323

Ответы:

427

Вы также можете использовать update ... fromсинтаксис и таблицу сопоставления. Если вы хотите обновить более одного столбца, это гораздо более обобщенно:

update test as t set
    column_a = c.column_a
from (values
    ('123', 1),
    ('345', 2)  
) as c(column_b, column_a) 
where c.column_b = t.column_b;

Вы можете добавить столько столбцов, сколько захотите:

update test as t set
    column_a = c.column_a,
    column_c = c.column_c
from (values
    ('123', 1, '---'),
    ('345', 2, '+++')  
) as c(column_b, column_a, column_c) 
where c.column_b = t.column_b;

sql fiddle demo

Роман Пекар
источник
11
Также может потребоваться указать правильный тип данных. Пример с датой: ... from (values ('2014-07-21'::timestamp, 1), ('2014-07-20', 2), ...Более подробная информация в документации PostgreSQL
Хосе Андиас
Прекрасно работает, спасибо за разъяснения! Документация Postgres по этому вопросу немного сбивает с толку.
skwidbreth
52

Основываясь на решении @Roman, вы можете установить несколько значений:

update users as u set -- postgres FTW
  email = u2.email,
  first_name = u2.first_name,
  last_name = u2.last_name
from (values
  (1, 'hollis@weimann.biz', 'Hollis', 'O\'Connell'),
  (2, 'robert@duncan.info', 'Robert', 'Duncan')
) as u2(id, email, first_name, last_name)
where u2.id = u.id;
Бенджамин Крузье
источник
4
Это похоже на его решение .. ОБНОВЛЕНИЕ ОТ (ЗНАЧЕНИЯ ...) ГДЕ. Как это только основано?
Эван Кэрролл
14
Я предпочитаю этот ответ, потому что имена переменных облегчают понимание того, что происходит.
Джон Леммон
Вот это да. Точно и ясно. Я пытаюсь реализовать что-то подобное в GoLang. Так я могу передать массив структур для значений? Примерно так: from (values $1)где $ 1 - это массив структур. В приведенном выше случае, строгий будет иметь свойства id, first_name и last_name.
Решма Суреш
26

Да, ты можешь:

UPDATE foobar SET column_a = CASE
   WHEN column_b = '123' THEN 1
   WHEN column_b = '345' THEN 2
END
WHERE column_b IN ('123','345')

И рабочее доказательство: http://sqlfiddle.com/#!2/97c7ea/1

zero323
источник
8
Это неправильно ... Вы обновите все строки, даже если это не так, '123'ни '345'. Вы должны использовать WHERE column_b IN ('123','456')...
MatheusOl
1
я думаю, что '456'должно быть'345'
Роман Пекар
2
Если вы добавите ELSE column_bпосле последней WHEN ? THEN ?строки, столбцу будет присвоено его текущее значение, что предотвратит то, что сказал MatheusQI.
Кевин Оррисс
1
Это не то, что он просил ... ему нужно обновить несколько столбцов, а не устанавливать столбец A на основе
столбца
Разве это не то, что запрашивал OP - только column_a нуждается в обновлении (на основе значения column_b), а не несколько столбцов, верно?
Кевлар
3

Наткнулся на похожий сценарий и выражение CASE было мне полезно.

UPDATE reports SET is_default = 
case 
 when report_id = 123 then true
 when report_id != 123 then false
end
WHERE account_id = 321;

Отчеты - это таблица здесь, account_id такой же, как для report_ids, упомянутых выше. Приведенный выше запрос установит для 1 записи (той, которая соответствует условию) значение true, а для всех несоответствующих - значение false.

Рикки Бой
источник
2

Для обновления нескольких строк в одном запросе, вы можете попробовать это

UPDATE table_name
SET 
column_1 = CASE WHEN any_column = value and any_column = value THEN column_1_value end,
column_2 = CASE WHEN any_column = value and any_column = value THEN column_2_value end,
column_3 = CASE WHEN any_column = value and any_column = value THEN column_3_value end,
.
.
.
column_n = CASE WHEN any_column = value and any_column = value THEN column_n_value end

если вам не нужны дополнительные условия, удалите andчасть этого запроса

Омар
источник
0

Допустим, у вас есть массив идентификаторов и эквивалентный массив статусов - вот пример, как это сделать со статическим SQL (SQL-запрос, который не изменяется из-за разных значений) массивов:

drop table if exists results_dummy;
create table results_dummy (id int, status text, created_at timestamp default now(), updated_at timestamp default now());
-- populate table with dummy rows
insert into results_dummy
(id, status)
select unnest(array[1,2,3,4,5]::int[]) as id, unnest(array['a','b','c','d','e']::text[]) as status;

select * from results_dummy;

-- THE update of multiple rows with/by different values
update results_dummy as rd
set    status=new.status, updated_at=now()
from (select unnest(array[1,2,5]::int[]) as id,unnest(array['a`','b`','e`']::text[]) as status) as new
where rd.id=new.id;

select * from results_dummy;

-- in code using **IDs** as first bind variable and **statuses** as the second bind variable:
update results_dummy as rd
set    status=new.status, updated_at=now()
from (select unnest(:1::int[]) as id,unnest(:2::text[]) as status) as new
where rd.id=new.id;
Таль Барда
источник