在CHARACTER VARYING字段中防止空字符串

Sim*_*yer 18 postgresql varchar constraints

我正在使用PostgreSQL,并希望阻止某些必需的CHARACTER VARYING(VARCHAR)字段允许空字符串输入.

这些字段还需要包含唯一值,因此我已经使用了一个唯一约束; 但是,这不会阻止原始(唯一)空值.

基本示例,其中用户名必须是唯一的而不是空的

| id | username | password |
+----+----------+----------+
| 1  | User1    | pw1      | #Allowed
| 2  | User2    | pw1      | #Allowed
| 3  | User2    | pw2      | #Already prevented by constraint
| 4  | ''       | pw2      | #Currently allowed, but needs to be prevented
Run Code Online (Sandbox Code Playgroud)

Fra*_*ens 27

使用检查约束:

CREATE TABLE foobar(
  x TEXT NOT NULL UNIQUE,
  CHECK (x <> '')
);

INSERT INTO foobar(x) VALUES('');
Run Code Online (Sandbox Code Playgroud)


Dmi*_*oda 5

定义表字段时,可以使用标准SQL'CONSTRAINT ... CHECK'子句:

CREATE TABLE test
(
    nonempty VARCHAR NOT NULL UNIQUE CONSTRAINT non_empty CHECK(length(nonempty)>0)
)
Run Code Online (Sandbox Code Playgroud)


wil*_*ser 5

作为一种特殊的约束,您可以将数据类型+约束放入一个 DOMAIN 中:

-- set search_path='tmp';

DROP DOMAIN birthdate CASCADE;
CREATE DOMAIN birthdate AS date DEFAULT NULL
    CHECK (value >= '1900-01-01' AND value <= now())
    ;

DROP DOMAIN username CASCADE;
CREATE DOMAIN username AS VARCHAR NOT NULL
    CHECK (length(value) > 0)
    ;

DROP TABLE employee CASCADE;
CREATE TABLE employee
    ( empno INTEGER NOT NULL PRIMARY KEY
    , dob birthdate
    , zname username
    , UNIQUE (zname)
    );
INSERT INTO employee(empno,dob,zname) 
  VALUES (1,'1980-02-02', 'John Doe' ), (2,'1980-02-02', 'Jon Doeh' );
INSERT INTO employee(empno,dob,zname)
  VALUES (3,'1980-02-02', '' ), (4,'1980-01-01', 'Joan Doh' );
Run Code Online (Sandbox Code Playgroud)

这将允许您一次又一次地重用域,而不必每次都复制约束。


-- 更新 2021-03-25(感谢@AlexanderPavlov)

Postgres 的实现似乎有一个严重的缺陷:可以标量子查询的结果中插入 NULL

下面的(无意义的)COALESCE()“修复”了这种行为。

这允许我们将数据库置于禁止状态。


\echo literal NULL

    INSERT INTO employee(empno,dob,zname) VALUES (5,'2021-02-02', NULL );

\echo empty (scalar) set

    INSERT INTO employee(empno,dob,zname) VALUES (6,'2021-02-02', (select zname from employee where 1=0) );

\echo empty COALESCE((scalar, NULL) ) set

    INSERT INTO employee(empno,dob,zname) VALUES (7,'2021-02-02', (select COALESCE(zname,NULL) from employee where 1=0) );

\echo empty set#2

    INSERT INTO employee(empno,dob,zname) (select 8,'2021-03-03', zname from employee where 1=0 );

\echo duplicate the complete table

    INSERT INTO employee(empno,dob,zname) (select 100+empno,dob+'1mon':: interval, upper(zname) from employee );

select * from employee;
Run Code Online (Sandbox Code Playgroud)

额外结果:


literal NULL
ERROR:  domain username does not allow null values
empty (scalar) set
INSERT 0 1
empty COALESCE((scalar, NULL) ) set
ERROR:  domain username does not allow null values
empty set#2
INSERT 0 0
duplicate the complete table
ERROR:  domain username does not allow null values
 empno |    dob     |  zname   
-------+------------+----------
     1 | 1980-02-02 | John Doe
     2 | 1980-02-02 | Jon Doeh
     6 | 2021-02-02 | 
(3 rows)
Run Code Online (Sandbox Code Playgroud)

  • PG 团队建议不要将 `NOT NULL` 添加到域中,而是将其添加到列中,请参阅“注释”部分 https://www.postgresql.org/docs/12/sql-createdomain.html (3认同)