How can I generate a unique string per record in a table in Postgres?

前端 未结 6 2051
栀梦
栀梦 2020-12-06 04:57

Say I have a table like posts, which has typical columns like id, body, created_at. I\'d like to generate a unique string with the creation of each post, for use in somethin

相关标签:
6条回答
  • 2020-12-06 05:03

    The easiest way probably to use the sequence to guarantee uniqueness (so after the seq add a fix x digit random number):

    CREATE SEQUENCE test_seq;
    CREATE TABLE test_table (
      id bigint NOT NULL DEFAULT (nextval('test_seq')::text || (LPAD(floor(random()*100000000)::text, 8, '0')))::bigint,
      txt TEXT
    );
    insert into test_table (txt) values ('1');
    insert into test_table (txt) values ('2');
    select id, txt from test_table;
    

    However this will waste a huge amount of records. (Note: the max bigInt is 9223372036854775807 if you use 8 digit random number at the end, you can only have 922337203 records. Thou 8 digit is probably not necessary. Also check the max number for your programming environment!)

    Alternatively you can use varchar for the id and even convert the above number with to_hex() or change to base36 like below (but for base36, try to not expose it to customer, in order to avoid some funny string showing up!):

    PostgreSQL: Is there a function that will convert a base-10 int into a base-36 string?

    0 讨论(0)
  • 2020-12-06 05:05

    Use a Feistel network. This technique works efficiently to generate unique random-looking strings in constant time without any collision.

    For a version with about 2 billion possible strings (2^31) of 6 letters, see this answer.

    For a 63 bits version based on bigint (9223372036854775808 distinct possible values), see this other answer.

    You may change the round function as explained in the first answer to introduce a secret element to have your own series of strings (not guessable).

    0 讨论(0)
  • 2020-12-06 05:06

    I don't claim the following is efficient, but it is how we have done this sort of thing in the past.

    CREATE FUNCTION make_uid() RETURNS text AS $$
    DECLARE
        new_uid text;
        done bool;
    BEGIN
        done := false;
        WHILE NOT done LOOP
            new_uid := md5(''||now()::text||random()::text);
            done := NOT exists(SELECT 1 FROM my_table WHERE uid=new_uid);
        END LOOP;
        RETURN new_uid;
    END;
    $$ LANGUAGE PLPGSQL VOLATILE;
    

    make_uid() can be used as the default for a column in my_table. Something like:

    ALTER TABLE my_table ADD COLUMN uid text NOT NULL DEFAULT make_uid();
    

    md5(''||now()::text||random()::text) can be adjusted to taste. You could consider encode(...,'base64') except some of the characters used in base-64 are not URL friendly.

    0 讨论(0)
  • 2020-12-06 05:08

    Check out a blog by Bruce. This gets you part way there. You will have to make sure it doesn't already exist. Maybe concat the primary key to it?

    Generating Random Data Via Sql

    "Ever need to generate random data? You can easily do it in client applications and server-side functions, but it is possible to generate random data in sql. The following query generates five lines of 40-character-length lowercase alphabetic strings:"

      SELECT
    (
      SELECT string_agg(x, '')
      FROM (
        SELECT chr(ascii('a') + floor(random() * 26)::integer)
        FROM generate_series(1, 40 + b * 0)
      ) AS y(x)
    )
    FROM generate_series(1,5) as a(b);
    
    0 讨论(0)
  • 2020-12-06 05:14

    All existing answers are WRONG because they are based on SELECT while generating unique index per table record. Let us assume that we need unique code per record while inserting: Imagine two concurrent INSERTs are happening same time by miracle (which happens very often than you think) for both inserts same code was generated because at the moment of SELECT that code did not exist in table. One instance will INSERT and other will fail.

    First let us create table with code field and add unique index

    CREATE TABLE my_table
    (
        code TEXT NOT NULL
    );
    
    CREATE UNIQUE INDEX ON my_table (lower(code));
    

    Then we should have function or procedure (you can use code inside for trigger also) where we 1. generate new code, 2. try to insert new record with new code and 3. if insert fails try again from step 1

    CREATE OR REPLACE PROCEDURE my_table_insert()
    AS $$
    DECLARE
        new_code TEXT;
    BEGIN
    
        LOOP
            new_code := LOWER(SUBSTRING(MD5(''||NOW()::TEXT||RANDOM()::TEXT) FOR 8));
            BEGIN
                INSERT INTO my_table (code) VALUES (new_code);
                EXIT;
            EXCEPTION WHEN unique_violation THEN
    
            END;
        END LOOP;
    
    END;
    $$ LANGUAGE PLPGSQL;
    

    This is guaranteed error free solution not like other solutions on this thread

    0 讨论(0)
  • 2020-12-06 05:15

    Use primary key in your data. If you really need alphanumeric unique string, you can use base-36 encoding. In PostgreSQL you can use this function.

    Example:

    select base36_encode(generate_series(1000000000,1000000010));
    
    GJDGXS
    GJDGXT
    GJDGXU
    GJDGXV
    GJDGXW
    GJDGXX
    GJDGXY
    GJDGXZ
    GJDGY0
    GJDGY1
    GJDGY2
    
    0 讨论(0)
提交回复
热议问题