受支持版本: 当前版本 (18) / 17 / 16 / 15 / 14
开发版本: devel

UPDATE

UPDATE — 更新表中的行

Synopsis

[ WITH [ RECURSIVE ] with_query [, ...] ]
UPDATE [ ONLY ] table_name [ * ] [ [ AS ] alias ]
    SET { column_name = { expression | DEFAULT } |
          ( column_name [, ...] ) = [ ROW ] ( { expression | DEFAULT } [, ...] ) |
          ( column_name [, ...] ) = ( sub-SELECT )
        } [, ...]
    [ FROM from_item [, ...] ]
    [ WHERE condition | WHERE CURRENT OF cursor_name ]
    [ RETURNING [ WITH ( { OLD | NEW } AS output_alias [, ...] ) ]
                { * | output_expression [ [ AS ] output_name ] } [, ...] ]

描述

UPDATE会更改所有满足条件的行中指定列 的值。只需在SET子句中提及需要修改的列; 未被显式修改的列会保留其原有值。

有两种方法可以利用数据库中其他表所包含的信息来修改一个表:使用子选择, 或者在FROM子句中指定附加表。哪种技术更合适取决 于具体情况。

可选的RETURNING子句使UPDATE 基于每个实际更新的行计算并返回一个或多个值。可以计算任何使用该表列和/或 FROM中提到的其他表列的表达式。默认使用该 表列的新值(更新后值),但也可以请求旧值(更新前值)。 RETURNING列表的语法与SELECT 的输出列表相同。

必须拥有该表上的UPDATE权限,或者至少拥有 待更新列的UPDATE权限。对于其值会在 expressions或者 condition中读取的任何列, 还必须拥有SELECT权限。

参数

with_query #

WITH子句允许你指定一个或多个子查询,这些子查 询可在UPDATE查询中按名称引用。详见 Section 7.8SELECT

table_name #

要更新的表名(可以是模式限定的)。如果在表名前指定了 ONLY,只会更新所提及表中的匹配行。如果未指定 ONLY,还会更新继承自该表的任何表中的匹配行。 可选地,可以在表名后指定*,以显式指示包含后代 表。

alias #

目标表的替代名称。提供别名时,它会完全隐藏该表的实际名称。 例如,给定UPDATE foo AS fUPDATE语句的其余部分必须将该表称为 f,而不是foo

column_name #

table_name命名 的表中的列名。如有需要,列名可以使用子字段名或数组下标 进行限定。指定目标列时不要包含表名 — 例如, UPDATE table_name SET table_name.col = 1是无 效的。

expression #

要赋给该列的表达式。该表达式可以使用该表中这一列和其他列的 旧值。

DEFAULT #

把该列设置为其默认值(如果没有为它指定特定的默认表达式,则该 值为 NULL)。标识列将被设置为由关联序列生成的新值。对于生成 列,允许指定这一项,但这仅仅指定了根据其生成表达式计算该列的 常规行为。

sub-SELECT #

一个SELECT子查询,其输出列数必须与它前面圆括号中 列列表的列数相同。执行时,该子查询返回的行数不得超过一行。如 果返回一行,则其列值会赋给目标列;如果不返回任何行,则把 NULL 值赋给目标列。该子查询可以引用正在更新的表的当前行的旧值。

from_item #

一个表表达式,允许其他表的列出现在WHERE条件 和更新表达式中。这里使用的语法与SELECT语句 的FROM子句相 同;例如,可以为表名指定别名。除非你打算进行自连接(这种情况 下它必须在from_item中带有别名出现), 否则不要把目标表重复写成from_item

condition #

一个返回boolean类型值的表达式。只有使这个表达式返 回true的行才会被更新。

cursor_name #

要在WHERE CURRENT OF条件中使用的游标名。 要被更新的行是最近一次从该游标中取出的那一行。该游标必须是针 对UPDATE目标表的非分组查询。注意, WHERE CURRENT OF不能与布尔条件同时指定。有 关将游标用于WHERE CURRENT OF的更多信息,请见 DECLARE

output_alias #

RETURNING列表中为OLDNEW行指定的可选替代名称。

默认情况下,可通过写 OLD.column_nameOLD.*返回目标表旧值,通过写 NEW.column_nameNEW.*返回新值。提供别名后,这些名称会被隐藏, 必须使用别名来引用旧行或新行。例如 RETURNING WITH (OLD AS o, NEW AS n) o.*, n.*

output_expression #

在每一行被更新后,由UPDATE命令计算并返回的 表达式。该表达式可以使用由table_name 命名的表或FROM中列出的表中的任何列名。写成 *可返回所有列。

列名或*可以用OLDNEW来限定,也可以用它们对应的 output_alias来限定, 以返回旧值或新值。未限定的列名或*,以及使用 目标表名或别名限定的列名或*,都会返回新值。

output_name #

用于返回列的名称。

输出

在成功完成时,一个UPDATE命令会返回以下形式 的命令标签:

UPDATE count

count是被更新的行数, 包括值没有改变的匹配行。注意,当更新被BEFORE UPDATE 触发器抑制时,这个数量可能小于匹配 condition的行数。如果 count为 0,则该查询没有更 新任何行(这不被视为错误)。

如果UPDATE命令包含RETURNING 子句,其结果将类似于一个SELECT语句,其中包含 RETURNING列表中定义的列和值,并在该命令更新的 行上进行计算。

注解

当存在FROM子句时,本质上会将目标表与 from_item列表中提到的表连接起来,而连接 的每一条输出行都代表对目标表的一次更新操作。使用 FROM时,应确保对每一个要修改的目标行,连接至多 生成一条输出行。换言之,一条目标行不应与其他表中的多于一行成功 连接。如果发生这种情况,则只会使用其中某一条连接行来更新目标行, 但具体使用哪一条并不容易预测。

因为存在这种不确定性,所以仅在子选择中引用其他表会更安全,尽管 这种写法通常比使用连接更难阅读、也更慢。

对于分区表,更新一行可能会导致它不再满足其所在分区的分区约束。 在这种情况下,如果在分区树中存在另一个满足该行分区约束的分区, 则该行会被移动到那个分区。如果不存在这样的分区,则会报错。在幕 后,行移动实际上是一次DELETE和一次 INSERT操作。

在被移动的行上并发执行UPDATEDELETE时,有可能收到序列化失败错误。假设会话 1 正在更新某个分区键,与此同时,一个对该行可见的并发会话 2 对该行 执行UPDATEDELETE操作。在 这种情况下,会话 2 的UPDATEDELETE将检测到行移动,并引发序列化失败错误 (其 SQLSTATE 代码始终为'40001')。如果发生 这种情况,应用程序可能需要重试事务。在表未分区或没有发生行移动 的通常情况下,会话 2 会识别出新更新的那一行,并在这个新行版本上执行 UPDATE/DELETE

请注意,虽然行可以从本地分区移动到外部表分区(前提是外部数据包装 器支持元组路由),但不能从外部表分区移动到另一个分区。

如果发现某个外键直接引用了源分区的某个祖先,并且这个祖先与 UPDATE查询中提到的祖先不是同一个,那么尝试把 一行从一个分区移动到另一个分区将会失败。

示例

把表filmskind 列中的单词Drama改为Dramatic

UPDATE films SET kind = 'Dramatic' WHERE kind = 'Drama';

在表weather的一行中调整温度项,并将降水量 重置为其默认值:

UPDATE weather SET temp_lo = temp_lo+1, temp_hi = temp_lo+15, prcp = DEFAULT
  WHERE city = 'San Francisco' AND date = '2003-07-03';

执行同样的操作,并返回更新后的各项以及旧的降水值:

UPDATE weather SET temp_lo = temp_lo+1, temp_hi = temp_lo+15, prcp = DEFAULT
  WHERE city = 'San Francisco' AND date = '2003-07-03'
  RETURNING temp_lo, temp_hi, prcp, old.prcp AS old_prcp;

使用另一种列列表语法完成同样的更新:

UPDATE weather SET (temp_lo, temp_hi, prcp) = (temp_lo+1, temp_lo+15, DEFAULT)
  WHERE city = 'San Francisco' AND date = '2003-07-03';

使用FROM子句语法,将负责 Acme Corporation 账户的 销售人员的销量计数加一:

UPDATE employees SET sales_count = sales_count + 1 FROM accounts
  WHERE accounts.name = 'Acme Corporation'
  AND employees.id = accounts.sales_person;

WHERE子句中使用子选择,执行同样的操作:

UPDATE employees SET sales_count = sales_count + 1 WHERE id =
  (SELECT sales_person FROM accounts WHERE name = 'Acme Corporation');

更新accounts表中的联系人姓名,使其与当前 分配的销售人员保持一致:

UPDATE accounts SET (contact_first_name, contact_last_name) =
    (SELECT first_name, last_name FROM employees
     WHERE employees.id = accounts.sales_person);

使用连接也可以得到类似结果:

UPDATE accounts SET contact_first_name = first_name,
                    contact_last_name = last_name
  FROM employees WHERE employees.id = accounts.sales_person;

但是,如果employees.id不是唯一键, 第二个查询可能会给出意外结果;而如果存在多个 id匹配,第一个查询则保证会报错。此外, 如果某个特定的accounts.sales_person 条目没有匹配项,第一个查询会把相应的姓名字段设为NULL, 而第二个查询则根本不会更新该行。

更新汇总表中的统计数据以匹配当前数据:

UPDATE summary s SET (sum_x, sum_y, avg_x, avg_y) =
    (SELECT sum(x), sum(y), avg(x), avg(y) FROM data d
     WHERE d.group_id = s.group_id);

尝试插入一个新库存项及其库存量。如果该项已存在,则改为更新现有 项的库存量。要在不致使整个事务失败的情况下做到这一点,请使用保存点:

BEGIN;
-- 其他操作
SAVEPOINT sp1;
INSERT INTO wines VALUES('Chateau Lafite 2003', '24');
-- 假设上面的语句因唯一键冲突而失败,
-- 那么现在发出这些命令:
ROLLBACK TO sp1;
UPDATE wines SET stock = stock + 24 WHERE winename = 'Chateau Lafite 2003';
-- 继续执行其他操作,最后
COMMIT;

更改表films中游标c_films 当前定位那一行的kind列:

UPDATE films SET kind = 'Dramatic' WHERE CURRENT OF c_films;

影响很多行的更新可能对系统性能产生负面影响,例如表膨胀、复制延 迟增加以及锁争用加剧。在这类场景中,将操作分成较小批次来执行可 能更有意义,并且可以在各批次之间对该表执行VACUUM。 虽然UPDATE没有LIMIT子句, 但可以通过公共表表达式和自连 接获得类似效果。对于标准的PostgreSQL 表访问方法,基于系统列ctid 的自连接非常高效:

WITH exceeded_max_retries AS (
  SELECT w.ctid FROM work_item AS w
    WHERE w.status = 'active' AND w.num_retries > 10
    ORDER BY w.retry_timestamp
    FOR UPDATE
    LIMIT 5000
)
UPDATE work_item SET status = 'failed'
  FROM exceeded_max_retries AS emr
  WHERE work_item.ctid = emr.ctid;

该命令需要重复执行,直到没有剩余行需要更新。 (这里使用ctid之所以安全,仅仅是因为查 询会重复运行,从而避免了ctid发生变化带 来的问题。) 使用ORDER BY子句可以让命令优先更新哪些行;如果 其他更新操作也使用相同的顺序,它还可以防止死锁。如果担心锁争 用,可以在CTE中加入SKIP LOCKED, 以防多个命令更新同一行。不过,这样仍需要在最后执行一次不带 SKIP LOCKEDLIMITUPDATE,以确保没有遗漏任何匹配行。

兼容性

这个命令符合SQL标准,不过 FROMRETURNING子句是 PostgreSQL扩展,把WITHUPDATE一起使用的能力也是扩展。

某些其他数据库系统提供一种FROM选项,要求在 FROM中再次列出目标表。PostgreSQL 并不是这样解释FROM的。在移植使用这种扩展的应 用时要小心。

根据标准,目标列名的一个圆括号子列表的源值可以是任何能够产生正 确列数的行值表达式。PostgreSQL只允许该 源值是一个行构造器 或子SELECT。在行构造器的情况下,单个列的更新值 可以指定为DEFAULT,但在子SELECT 中则不能这样做。

提交更正

如果您发现文档中有不正确的内容、与您使用特定功能的经验不符或需要进一步说明,请使用此表单来报告文档问题。