Добавил:
Upload Опубликованный материал нарушает ваши авторские права? Сообщите нам.
Вуз: Предмет: Файл:
Основы SQL-Курс лекций ИНТУИТ.docx
Скачиваний:
180
Добавлен:
16.09.2019
Размер:
554.17 Кб
Скачать

Пример 14.4. Триггер для обработки операции изменения записи в таблице

В рассмотренном триггере отсутствует сравнение количества товара при изменении записи о сделке с его остатком на складе.

Пример 14.5. Исправим этот недостаток. Для генерирования сообщения об ошибке используем в теле триггера команду MS SQL Server RAISERROR, аргументами которой являются текст сообщения, уровень серьезности и статус ошибки.

ALTER TRIGGER Триггер_upd

ON Сделка FOR UPDATE

AS

DECLARE @x INT, @x_old INT, @y INT,

@y_old INT ,@o INT

DECLARE CUR1 CURSOR FOR

SELECT КодТовара,Количество

FROM inserted

DECLARE CUR2 CURSOR FOR

SELECT КодТовара,Количество

FROM deleted

OPEN CUR1

OPEN CUR2

FETCH NEXT FROM CUR1 INTO @x, @y

FETCH NEXT FROM CUR2 INTO @x_old, @y_old

WHILE @@FETCH_STATUS=0

BEGIN

SELECT @o=остаток

FROM Склад

WHERE кодтовара=@x

IF @o<-@y

BEGIN

RAISERROR('откат',16,10)

CLOSE CUR1

CLOSE CUR2

DEALLOCATE CUR1

DEALLOCATE CUR2

ROLLBACK TRAN

RETURN

END

UPDATE Склад

SET Остаток=Остаток-@y_old

WHERE КодТовара=@x_old

IF NOT EXISTS (SELECT * FROM Склад

WHERE КодТовара=@x)

INSERT INTO Склад(КодТовара,Остаток)

VALUES (@x,@y)

ELSE

UPDATE Склад

SET Остаток=Остаток+@y

WHERE КодТовара=@x

FETCH NEXT FROM CUR1 INTO @x, @y

FETCH NEXT FROM CUR2 INTO @x_old, @y_old

END

CLOSE CUR1

CLOSE CUR2

DEALLOCATE CUR1

DEALLOCATE CUR2

Пример 14.5. Исправленный вариант триггера для обработки операции изменения записи в таблице

Пример 14.6. В примере 14.5 происходит отмена всех изменений при невозможности реализовать хотя бы одно из них. Создадим триггер, позволяющий отменять изменение только некоторых записей и выполнять изменение остальных.

В этом случае триггер выполняется не после изменения записей, а вместо команды изменения.

ALTER TRIGGER Триггер_upd

ON Сделка INSTEAD OF UPDATE

AS

DECLARE @k INT, @k_old INT

DECLARE @x INT, @x_old INT, @y INT

DECLARE @y_old INT ,@o INT

DECLARE CUR1 CURSOR FOR

SELECT КодСделки, КодТовара,Количество

FROM inserted

DECLARE CUR2 CURSOR FOR

SELECT КодСделки, КодТовара,Количество

FROM deleted

OPEN CUR1

OPEN CUR2

FETCH NEXT FROM CUR1 INTO @k,@x, @y

FETCH NEXT FROM CUR2 INTO @k_old,@x_old,

@y_old

WHILE @@FETCH_STATUS=0

BEGIN

SELECT @o=остаток

FROM Склад

WHERE КодТовара=@x

IF @o>=-@y

BEGIN

RAISERROR('изменение',16,10)

UPDATE Сделка SET количество=@y,

КодТовара=@x

WHERE КодСделки=@k

UPDATE Склад

SET Остаток=Остаток-@y_old

WHERE КодТовара=@x_old

IF NOT EXISTS (SELECT * FROM Склад

WHERE КодТовара=@x)

INSERT INTO Склад(КодТовара, Остаток)

VALUES (@x,@y)

ELSE

UPDATE Склад

SET Остаток=Остаток+@y

WHERE КодТовара=@x

END

ELSE

RAISERROR('запись не изменена',16,10)

FETCH NEXT FROM CUR1 INTO @k,@x, @y

FETCH NEXT FROM CUR2 INTO @k_old,@x_old,

@y_old

END

CLOSE CUR1

CLOSE CUR2

DEALLOCATE CUR1

DEALLOCATE CUR2

Пример 14.6. Триггер, позволяющий отменять изменение только некоторых записей и выполнять изменение остальных.

Лекция 15: Триггеры в рекурсивных структурах Дается определение рекурсивной структуры в реляционной базе данных. Приводится пример создания таблицы, реализующей рекурсивную иерархию. Определяются правила целостности, непротиворечивости и достоверности данных в таблицах с рекурсивными связями. Описывается программирование триггеров, поддерживающих целостность данных в рекурсивных структурах.

Введение в рекурсивные структуры

Рассмотрим создание таблицы, реализующей рекурсивную иерархию, на примере данных, описывающих отношения подчиненности между сотрудниками. В таблице emp_mgrнеобходимо задать как имя сотрудника ( emp ), так и имя его начальника ( mgr ). Для рекурсивной связи одна и та же сущность является и родительской, и дочерней. При задании рекурсивной связи атрибут первичного ключа мигрирует в качестве внешнего ключа в состав неключевых атрибутов той же сущности (атрибуты emp – сотрудник иmgr – начальник таблицы emp_mgr ). Информация о руководителе содержится в той же сущности, поскольку руководитель – сотрудник той же организации. Связь руководит/подчиняется ( fk_emp ) позволяет хранить древовидную иерархию подчиненности. Такой вид рекурсивной связи называется иерархической рекурсией и задает связь, когда руководитель (экземпляр родительской сущности) может иметь множество подчиненных (экземпляров дочерней сущности), но подчиненный – только одного руководителя. В среде MS SQL Server создадим таблицу emp_mgr:

CREATE TABLE emp_mgr

(emp CHAR(2) PRIMARY KEY,

mgr CHAR(2) NULL,

NoOfReports INT DEFAULT 0,

CONSTRAINT fk_emp FOREIGN KEY (mgr)

REFERENCES emp_mgr (emp) )

В таблицу введено поле NoOfReports, в котором для каждого сотрудника определено количество его подчиненных.

Для удобства иллюстрации в качестве имени сотрудника и его начальника будут использоваться латинские буквы. Например, ввод данных в таблицу осуществляется операторами:

INSERT INTO emp_mgr(emp,mgr) VALUES('a',NULL)

INSERT INTO emp_mgr(emp,mgr) VALUES('b','a')

INSERT INTO emp_mgr(emp,mgr) VALUES('c','a')

INSERT INTO emp_mgr(emp,mgr) VALUES('d','a')

INSERT INTO emp_mgr(emp,mgr) VALUES('e','b')

INSERT INTO emp_mgr(emp,mgr) VALUES('f','b')

INSERT INTO emp_mgr(emp,mgr) VALUES('g','b')

INSERT INTO emp_mgr(emp,mgr) VALUES('i','c')

INSERT INTO emp_mgr(emp,mgr) VALUES('k','d')

После ввода данных в таблицу emp_mgr оператор SELECT * FROM emp_mgr возвращает следующий результат:

emp mgr NoOfReports

-------------------------

a NULL 3

b a 3

c a 1

d a 1

e b 0

f b 0

g b 0

i c 0

k d 0