如何在for循环中触发更新值,Postgresql?

时间:2017-04-24 10:20:21

标签: sql postgresql plpgsql database-trigger

我有包含任务的表。 每项任务都有优先权。 当我插入一个新任务时,如果新任务与旧任务具有相同的优先级,那么具有较低优先级的旧任务必须是低级别的。

例如,我们有3个优先级任务:1,2,3。如果我插入优先级为1的新任务,则所有旧任务必须在插入新任务之前将优先级更改为2,3,4。

为了做到这一点,我使用了触发器:

CREATE SEQUENCE task_id_seq;

CREATE TABLE task 
(
    task_id INTEGER NOT NULL DEFAULT nextval('task_id_seq'),
    name VARCHAR(255) NOT NULL,
    description TEXT,
    created_time TIMESTAMP WITH TIME ZONE NOT NULL DEFAULT CURRENT_TIMESTAMP,
    deadline TIMESTAMP WITH TIME ZONE,
    priority INT2 NOT NULL,
    PRIMARY KEY (task_id),
    CONSTRAINT task__check_task_name CHECK(CHAR_LENGTH(name) >= 4),
    CONSTRAINT task__check_unique_task_name UNIQUE(name),
    CONSTRAINT task__check_unique_task_priority UNIQUE(priority)
);

CREATE INDEX priority_index ON task(priority);

CREATE OR REPLACE FUNCTION degrade_task_priorities()
RETURNS trigger AS $degrade_priority_task_trigger$
DECLARE
    r task%rowtype;
BEGIN
    FOR r in SELECT * FROM task WHERE task.priority >= NEW.priority
    LOOP
        --r.priority := r.priority + 1;
        UPDATE task SET priority = priority + 1;
        --WHERE task.priority In (
        --  SELECT priority FROM task WHERE task.priority >= NEW.priority ORDER BY task.priority);
        --RETURN NEXT r;
    END LOOP;
    RETURN NEW;
END
$degrade_priority_task_trigger$ LANGUAGE plpgsql;


CREATE TRIGGER insert_new_task_trigger
BEFORE INSERT ON task
FOR EACH ROW
EXECUTE PROCEDURE degrade_task_priorities();

问题是当我尝试插入4个具有相同优先级1的新任务时,最终结果是4个优先级如下的记录:1,4,6,7。 我应该是1,2,3,4。

我怀疑我不理解循环和触发器一起工作的方式。 我在哪里弄错了?

1 个答案:

答案 0 :(得分:1)

我相信你不需要循环:

CREATE OR REPLACE FUNCTION degrade_task_priorities()
RETURNS trigger AS $degrade_priority_task_trigger$
DECLARE
    r task%rowtype;
BEGIN
    UPDATE task SET priority = priority + 1 WHERE task.priority >= NEW.PRIORITY;
    RETURN NEW;
END
$degrade_priority_task_trigger$ LANGUAGE plpgsql;