Ограничения CHECK
Ограничения CHECK
Ограничение CHECK используется для проверки достоверности вводимых значений данных. Оно реализует условия или требования, которым должно удовлетворять значение для успешного добавления или обновления. Оно не может изменять вводимое значение; оно лишь вернет исключение проверки достоверности, если значение не будет соответствовать условию.
! ! !
СОВЕТ. Ограничения CHECK применяются после выполнения триггеров "before". Используйте триггер, если вы хотите выполнить проверку и присвоить данным допустимые значения.
. ! .
В определении таблицы это ограничение на уровне таблицы. В отличие от ограничений CHECK, применимых к определению домена, его элемент VALUE является ссылкой на столбец. Например, для домена предложение CHECK может быть следующим:
CHECK(VALUE > 10)
В определении таблицы то же самое условие для столбца с именем ACOLUMN будет представлено как:
CHECK (ACOLUMN > 10)
Ограничение CHECK будет активным для операций INSERT и UPDATE. Хотя это ограничение на уровне таблицы, его область действия может выходить за пределы уровня столбца на уровень строки и, хотя это не рекомендуется, на уровень таблицы и даже за пределы таблицы. Ограничение CHECK гарантирует целостность данных, только если значения, включенные в проверку, находятся в той же самой строке, что и проверяемое значение.
! ! !
ВНИМАНИЕ! Вы не должны использовать выражения, которые сравнивают значение со значениями из других строк этой же таблицы или других таблиц, потому что любая строка, отличная от текущей, может находиться в процессе изменения или удаления в другой транзакции. В особенности не полагайтесь на ограничение CHECK для проверки ссылочных отношений!
. ! .
Условие поиска может:
* проверять, что вводимое значение находится в указанном диапазоне;
* проверять, что значение присутствует в списке допустимых значений;
* сравнивать значение с константой, выражением или со значением данных другого столбца из этой же строки.
Замечания по использованию CHECK
Существуют определенные условия использования ограничений CHECK.
* Столбец может иметь только одно ограничение CHECK, хотя его логика может быть представлена как сложное условие поиска - одно ограничение, много условий.
* Ограничение CHECK для столбца, основанного на домене, не может перекрывать наследуемую от домена проверку. Определение столбца может использовать обычное предложение CHECK, чтобы добавить дополнительную логику ограничения в наследуемое ограничение. Оно будет соединено с наследуемым ограничением конъюнкцией (логической операцией И).
* Ограничение CHECK не может ссылаться на домен. Синтаксис ограничения CHECK:
CHECK (<условие-поиска>);
<условие-поиска> =
<val> <operator> {<val> | {<выбор-одного>) }
| <val> [NOT] BETWEEN <val> AND <val>
| <val> [NOT] LIKE <val> [ESCAPE <val>]
| <val> [NOT] IN (<val> [, <val> ...] | <список-выбора>)
| <val> IS [NOT] NULL
| <val> { [NOT] { = | < | > } |>=|<=}
{ ALL | SOME | ANY }(<список-выбора>)
| EXISTS (<выражение-выбора>)
| SINGULAR (<выражение-выбора>)
| <val> [NOT] CONTAINING <val>
| <val> [NOT] STARTING [WITH] <val>
| (<условие-поиска>)
| NOT <условие-поиска>
| <условие-поиска> OR <условие-поиска>
l <условие-поиска> AND <условие-поиска>
Диапазон возможностей при определении ограничений CHECK действительно весьма широк - теоретически в нем может быть использовано почти любое условие поиска. Для разработчика важно выбрать разумные и безопасные условия, т. к. они действуют для каждой операции INSERT и UPDATE для таблицы.
В части V данной книги описывается синтаксис установки различных стилей условия поиска.
Пример. Следующее ограничение проверяет значения двух столбцов для гарантии, что одно больше другого. Хотя оно также предполагает условия NOT NULL для обоих столбцов - проверка будет ошибочной, если хотя бы один столбец будет иметь пустое значение, - оно не устанавливает ограничение NOT NULL для столбца:
CHECK (COL_1 > COL_2);
Проверка даст ошибку, если арифметическая проверка будет ошибочной или если любой из столбцов COL_1 или COL_2 имеет значение NULL. Следующая операция будет успешной:
INSERT INTO TABLE_1 (COL_1, COL_2) VALUES (6,5);