В системных таблицах RDB$PROCEDURES
, RDB$FUNCTIONS
и RDB$TRIGGERS
присутствует поле RDB$VALID_BLR
.
Оно предназначено для сигнализации о возможной недействительности модуля PSQL (процедуры или триггера) после изменения доменов или столбцов таблиц, от которых он зависит.
При возникновении описанной выше ситуации поле RDB$VALID_BLR
устанавливается в 0 для процедур или триггеров, код которых возможно является недействительным.
В триггерах, процедурах и функциях зависимости возникают от столбцов таблицы, к которой они обращаются, а так же от любого параметра или переменной, определенных в модуле с использованием предложения TYPE OF
.
После того как ядро Firebird изменило любой домен, включая неявные домены, создаваемые внутри при определении столбцов или параметров, Firebird производит внутреннюю перекомпиляцию всех зависимостей.
Note
|
Инвалидация происходит для процедур, функций, пакетов и триггеров, но не для блоков DML операторов, которые выполняются при помощи EXECUTE BLOCK. |
Любой модуль, который не удалось перекомпилировать из-за несовместимости, возникающей из-за изменения домена, помечается как недействительный (поле RDB$VALID_BLR
устанавливается в 0 в записи соответствующей системной таблице (RDB$PROCEDURES
или RDB$TRIGGERS
).
Возобновление (установка RDB$VALID_BLR
в 1) происходит когда
-
домен изменён снова и его новое определение совместимо с ранее недействительным определением модуля; или
-
ранее недействительный модуль изменён так, что соответствовать новому определению домена.
Нижеприведённый запрос находит процедуры и триггеры, зависящие от определённого домена (в примере это домен 'MYDOMAIN'), и выводит информацию о состоянии поля RDB$VALID_BLR
:
WITH VALID_PSQL (
PSQL_TYPE,
ROUTE_NAME,
VALID)
AS (SELECT
'Procedure',
RDB$PROCEDURE_NAME,
RDB$VALID_BLR
FROM
RDB$PROCEDURES
WHERE
RDB$PROCEDURES.RDB$PACKAGE_NAME IS NULL
UNION ALL
SELECT
'Function',
RDB$FUNCTION_NAME,
RDB$VALID_BLR
FROM
RDB$FUNCTIONS
WHERE
RDB$FUNCTIONS.RDB$PACKAGE_NAME IS NULL
UNION ALL
SELECT
'Package',
RDB$PACKAGE_NAME,
RDB$VALID_BODY_FLAG
FROM
RDB$PACKAGES
UNION ALL
SELECT
'Trigger',
RDB$TRIGGER_NAME,
RDB$VALID_BLR
FROM
RDB$TRIGGERS
WHERE
RDB$TRIGGERS.RDB$SYSTEM_FLAG = 0)
SELECT
PSQL_TYPE,
ROUTE_NAME,
VALID
FROM
VALID_PSQL
WHERE
EXISTS(SELECT
*
FROM
RDB$DEPENDENCIES
WHERE
RDB$DEPENDENT_NAME = VALID_PSQL.ROUTE_NAME
AND RDB$DEPENDED_ON_NAME = 'MYDOMAIN');
/*
Замените MYDOMAIN фактическим именем проверяемого
домена. Используйте заглавные буквы, если
домен создавался нечувствительным к регистру — в
противном случае используйте точное написание
имени домена с учётом регистра
*/
Следующий запрос находит процедуры и триггеры, зависящие от определённого столбца таблицы (в примере это столбец 'MYCOLUMN' таблицы 'MYTABLE'), и выводит информацию о состоянии поля RDB$VALID_BLR
:
WITH VALID_PSQL (
PSQL_TYPE,
ROUTE_NAME,
VALID)
AS (SELECT
'Procedure',
RDB$PROCEDURE_NAME,
RDB$VALID_BLR
FROM
RDB$PROCEDURES
WHERE
RDB$PROCEDURES.RDB$PACKAGE_NAME IS NULL
UNION ALL
SELECT
'Function',
RDB$FUNCTION_NAME,
RDB$VALID_BLR
FROM
RDB$FUNCTIONS
WHERE
RDB$FUNCTIONS.RDB$PACKAGE_NAME IS NULL
UNION ALL
SELECT
'Package',
RDB$PACKAGE_NAME,
RDB$VALID_BODY_FLAG
FROM
RDB$PACKAGES
UNION ALL
SELECT
'Trigger',
RDB$TRIGGER_NAME,
RDB$VALID_BLR
FROM
RDB$TRIGGERS
WHERE
RDB$TRIGGERS.RDB$SYSTEM_FLAG = 0)
SELECT
PSQL_TYPE,
ROUTE_NAME,
VALID
FROM
VALID_PSQL
WHERE
EXISTS(SELECT
*
FROM
RDB$DEPENDENCIES D
WHERE
D.RDB$DEPENDENT_NAME = VALID_PSQL.ROUTE_NAME
AND D.RDB$DEPENDED_ON_NAME = 'MYTABLE'
AND D.RDB$FIELD_NAME = 'MYCOLUMN');
/*
Замените MYTABLE и MYCOLUMN фактическими именами
проверяемой таблицы и её столбца.
Используйте заглавные буквы, если таблица и её
столбец создавались нечувствительными к регистру —
в противном случае используйте точное написание
имени таблицы и её столбца с учётом регистра
*/
Important
|
Все случаи возникновения недействительных модулей, вызванных изменениями доменов/столбцов, отражаются в поле
|
Note
|
Другие замечания
|
Important
|
Это замечание об операторах равенства и неравенства применяется повсюду в СУБД Firebird. |
Оператор “=”, который используется во многих условиях, сравнивает только значения со значениями.
В соответствии со стандартом SQL, NULL
не является значением и, следовательно, два значения NULL
не равны и ни неравны друг с другом.
Если необходимо, чтобы значения NULL
соответствовали друг другу при объединении, используйте оператор IS NOT DISTINCT FROM
.
Этот оператор возвращает истину, если операнды имеют то же значение, или, если оба они равны NULL
.
SELECT *
FROM A
JOIN B ON A.id IS NOT DISTINCT FROM B.code
Точно так же, если вы хотите чтобы значения NULL
отличались от любого значения и два значения NULL
считались равными, используйте оператор IS DISTINCT FROM
вместо оператора “<>”.
SELECT *
FROM A
JOIN B ON A.id IS DISTINCT FROM B.code