更新数据库行而不在PostgreSQL 9.2中锁定表
尝试使用PostgreSQL 9.2在表上运行这样的更新语句: UPDATE table SET a_col = array[col]; 我们需要能够在~10M的行表上运行它,而不是让它锁定表(因此在更新运行时仍然可以进行正常操作).我相信使用游标可能是正确的解决方案,但我真的不知道它是否或我应该如何使用游标实现它. 我想出了这个游标代码,我认为这可能是好的. 编辑:添加了光标功能 CREATE OR REPLACE FUNCTION update_fields() RETURNS VOID AS $$ DECLARE cursor CURSOR FOR SELECT * FROM table ORDER BY id FOR UPDATE; BEGIN FOR row IN cursor LOOP UPDATE table SET a_col = array[col],a_col2= array[col2] WHERE CURRENT OF cursor; END LOOP; END; $$LANGUAGE plpgsql; 解决方法MVCC首先,如果“正常操作”由SELECT查询组成,MVCC model将自动处理它. UPDATE不会阻止SELECT,反之亦然. SELECT仅查看已提交的数据(或在同一事务中已完成的操作),因此大UPDATE的结果对其他事务保持不可见,直到完成(已提交). 表现/膨胀 如果您没有引用该表的其他对象, CREATE TABLE tbl_new (LIKE tbl_org INCLUDING CONSTRAINTS); INSERT INTO tbl_new SELECT col_a,col_b,array[col] aS col_c FROM tbl_org; 我正在使用CREATE TABLE(LIKE ..包括CONSTRAINTS),因为(quoting the manual here):
确保新表已准备就绪.然后: DROP tbl_org; ALTER TABLE tbl_new RENAME TO tbl_org; 结果在一个非常短的时间窗口中,表格被锁定. 这实际上只是关于性能.它创建了一个没有任何臃肿的新表.如果您有外键或视图,您仍然可以使用该路径,但您必须准备一个脚本来删除并重新创建这些对象,从而可能创建其他独占锁. 并发写道 通过并发写入操作,您可以做的就是将更新分成块.您不能在单个事务中执行此操作,因为锁仅在事务结束时释放. 您可以使用dblink,它可以在另一个数据库上启动独立事务,包括它自己.这样你就可以在单个DO语句或带循环的plpgsql函数中完成所有操作.这是一个松散相关的答案,有关dblink的更多信息: > Drop or create database from stored procedure in PostgreSQL 你使用游标的方法 函数内的光标不会给你买任何东西.任何函数都自动包含在事务中,并且所有锁仅在事务结束时释放.
您需要运行单独的事务或(ab)使用dblink为您执行此操作. (编辑:PHP编程网 - 黄冈站长网) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |