Autoincrement, but omit existing values in the column

前端 未结 3 1504
孤独总比滥情好
孤独总比滥情好 2021-01-24 11:46

I have a table:

create table DB.t1 (id  SERIAL,name varchar(255));

and insert some data:

insert into DB.t1 (name) values (\'nam         


        
3条回答
  •  终归单人心
    2021-01-24 12:43

    Update: Later, more detailed answer:

    • Make Postgres choose the next minimal available id

    This should work smoothly:

    CREATE OR REPLACE FUNCTION f_next_free(_seq regclass)
      RETURNS integer AS
    $func$
    BEGIN
       LOOP
          PERFORM nextval(_seq);
          EXIT WHEN NOT EXISTS (SELECT 1 FROM db.t1 WHERE id = lastval());
       END LOOP; 
    
       RETURN lastval();
    END
    $func$  LANGUAGE plpgsql VOLATILE;
    

    The loop is fetching the next number from the given sequence until one is found that is not yet in the table. Should even be safe for concurrent use, since we still rely on a sequence.

    Use this function in the column default of the serial column (replacing the default for the serial columns nextval('t1_id_seq'::regclass):

    ALTER TABLE db.t1 ALTER COLUMN id
    SET DEFAULT f_next_free('t1_id_seq'::regclass);
    

    The manual on lastval().

    This performs well with few islands and many gaps (which seems to be the case according to the example). To enforce uniqueness, add a unique constraint (or primary key) on the column.

提交回复
热议问题