CREATE DOMAIN
CREATE DOMAIN — создать домен
Синтаксис
CREATE DOMAINимя
[ AS ]тип_данных
[ COLLATEправило_сортировки
] [ DEFAULTвыражение
] [ограничение_домена
[ ... ] ] Здесьограничение_домена
: [ CONSTRAINTимя_ограничения
] { NOT NULL | NULL | CHECK (выражение
) }
Описание
CREATE DOMAIN
создаёт новый домен. Домен по сути представляет собой тип данных с дополнительными условиями (ограничивающими допустимый набор значений). Владельцем домена становится пользователь его создавший.
Если задаётся имя схемы (например, CREATE DOMAIN myschema.mydomain ...
), домен создаётся в указанной схеме, в противном случае — в текущей. Имя домена должно быть уникальным среди имён типов и доменов, существующих в этой схеме.
Домены полезны для абстрагирования и вынесения общих характеристик разных полей в единое место для упрощения сопровождения. Например, в нескольких таблицах может присутствовать столбец, содержащий электронный адрес, и для всех требуются одинаковые ограничения CHECK, проверяющие синтаксис адреса. В этом случае лучше определить домен, а не задавать для каждой таблицы отдельные ограничения.
Чтобы создать домен, необходимо иметь право USAGE
для нижележащего типа.
Параметры
имя
Имя создаваемого домена (возможно, дополненное схемой).
тип_данных
Нижележащий тип данных домена (может включать определение массива с этим типом).
правило_сортировки
Необязательное указание правила сортировки для домена. Если это указание отсутствует, в домене используется правило сортировки нижележащего типа данных. Указать
COLLATE
можно, только если нижележащий тип данных является сортируемым.DEFAULT
выражение
Предложение
DEFAULT
определяет значение по умолчанию для столбцов, типом данных которых является этот домен. Значением может быть любое выражение без переменных (подзапросы также не допускаются). Тип данных этого выражения должен соответствовать типу данных домена. Если значение по умолчанию не указано, им будет значение NULL.Значение по умолчанию будет использоваться в любой операции добавления строк, в которой не задано значение для этого столбца. Если значение по умолчанию установлено для конкретного столбца, оно будет переопределять значение по умолчанию, связанное с доменом. В свою очередь, значение по умолчанию для домена переопределяет любое значение по умолчанию, связанное с нижележащим типом данных.
CONSTRAINT
имя_ограничения
Имя ограничения. Если не указано явно, имя будет сгенерировано системой.
NOT NULL
Значения этого домена будут отличны от NULL (но см. замечания ниже).
NULL
Этот домен может содержать значение NULL. Это свойство домена по умолчанию.
Это предложение предназначено только для совместимости с нестандартными базами данных SQL. Использовать его в новых приложениях не рекомендуется.
CHECK (
выражение
)Предложения
CHECK
задают ограничения целостности или проверки, которым должны удовлетворять значения домена. Каждое ограничение должно представлять собой выражение, выдающее результат типа Boolean. Проверяемое значение в этом выражении обозначается ключевым словомVALUE
. Если выражение выдаёт FALSE, сообщается об ошибке и приведение значения к типу домена запрещается.В настоящее время выражения
CHECK
не могут содержать переменные, кромеVALUE
, и подзапросы.Когда для домена задано несколько ограничений
CHECK
, они будут проверяться в алфавитном порядке имён. (До версии 9.5 в PostgreSQL не было установлено никакого определённого порядка обработки ограниченийCHECK
.)
Примечания #
Ограничения домена, в частности NOT NULL
, проверяются при преобразовании значения к типу домена. Однако из столбца, который номинально имеет тип домена, всё же можно прочитать NULL, несмотря на такое ограничение. Например, это может происходить в запросе внешнего соединения, если столбец домена окажется в обнуляемой стороне внешнего соединения. Более тонкий пример:
INSERT INTO tab (domcol) VALUES ((SELECT domcol FROM tab WHERE false));
Пустой скалярный вложенный SELECT выдаст значение NULL, типом которого будет считаться домен, так что к этому значению не будут применены дополнительные проверки ограничений и строка будет успешно добавлена.
Избежать таких проблем очень сложно, так как в SQL вообще предполагается, что значение NULL является подходящим для любого типа данных. Таким образом, лучше всего разрабатывать ограничения так, чтобы значения NULL допускались, а затем при необходимости применять ограничения NOT NULL
к столбцам доменного типа, а не непосредственно к самому этому типу.
В Postgres Pro предполагается, что условия ограничений CHECK
являются постоянными, то есть при одинаковых входных значениях они всегда выдают одинаковый результат. Именно этим предположением оправдывается то, что ограничения CHECK
проверяются только при первом преобразовании значения в тип домена, а не при каждом обращении к нему. (По сути таким же образом обрабатываются ограничения CHECK
для таблиц, как описано в Подразделе 5.5.1.)
Однако это предположение может нарушаться, как часто бывает, когда в выражении CHECK
используется пользовательская функция, поведение которой впоследствии меняется. Postgres Pro не запрещает этого, и если сохранённые значения типа домена перестанут удовлетворять ограничению CHECK
, это останется незамеченным. В итоге при попытке загрузить выгруженные позже данные могут возникнуть проблемы. Поэтому подобные изменения рекомендуется осуществлять следующим образом: удалить ограничение (используя ALTER DOMAIN
), изменить определение функции, а затем пересоздать ограничение той же командой, которая при этом перепроверит сохранённые данные.
Рекомендуется следить за тем, чтобы выражения CHECK
не выдавали ошибки.
Примеры
В этом примере создаётся тип данных us_postal_code
(почтовый индекс США), который затем используется в определении таблицы. Для проверки значения на соответствие формату почтовых индексов США применяется проверка с регулярными выражениями:
CREATE DOMAIN us_postal_code AS TEXT CHECK( VALUE ~ '^\d{5}$' OR VALUE ~ '^\d{5}-\d{4}$' ); CREATE TABLE us_snail_addy ( address_id SERIAL PRIMARY KEY, street1 TEXT NOT NULL, street2 TEXT, street3 TEXT, city TEXT NOT NULL, postal us_postal_code NOT NULL );
Совместимость #
Команда CREATE DOMAIN
соответствует стандарту SQL.
Синтаксис NOT NULL
данной команды является расширением Postgres Pro (аналог CHECK (VALUE IS NOT NULL)
соответствует стандарту; однако в соответствии с Разделом «Примечания» подобных ограничений лучше избегать). «Ограничение» NULL
является расширением Postgres Pro (см. также Совместимость).