Страница 4 из 6 Удаление одиночного узла в середине дерева
Удаление одиночного узла в середине дерева тяжелее чем удаление полных поддеревьев. Когда Вы удаляете узел в середине дерева, Вы должны решить, как заполнить отверстие. Имеются два пути. Первый метод к повышает одного из детей к позиции первоначального узла (предположим, что отец умирает, и самый старший сын занимает бизнес, как показано на рисунке 2). Самый старший потомок всегда показывается как крайний левый дочерний узел под родителем. Однако с этой этой операцией имеется проблема. Если старший ребенок имеет собственнх детей, Вы должны решить, как обработать их, и так далее вниз по дереву, пока Вы не добираетесь к листу. Второй метод для удаления одиночного узла в середине дерева состоит в том, чтобы подключить потомков к предку первоначального узла (можно сказать, что мамочка умирает, и дети приняты бабушкой), как показано на рисунке 3. Это получается автоматически во модели вложенных множеств, Вы просто удаляете узел, и его дети уже содержатся в узлах его предка. Однако, Вы должны быть осторожны, при закрытии промежутка, оставленного стиранием. Имеется различие в изменении нумерации потомков удаленного узла и изменения нумерации узлов справа. Ниже - процедура выполняющая это: CREATE PROCEDURE DropNode (downsized IN CHAR(10) NOT NULL) BEGIN ATOMIC DECLARE dropemp CHAR(10) NOT NULL; DECLARE droplft INTEGER NOT NULL; DECLARE droprgt INTEGER NOT NULL;
--Теперь сохраним данные поддерева:
SELECT emp, lft, rgt INTO dropemp, droplft, droprgt FROM Personnel WHERE emp = downsized;
--Удаление, это просто...
DELETE FROM Personnel WHERE emp = downsized;
--Теперь уплотняем промежутки:
UPDATE Personnel SET lft = CASE WHEN lft BETWEEN droplft AND droprgt THEN lft - 1 WHEN lft > droprgt THEN lft - 2 ELSE lft END rgt = CASE WHEN rgt BETWEEN droplft AND droprgt THEN rgt - 1 WHEN rgt > droprgt THEN rgt -2 ELSE rgt END; WHERE lft > droplft; END;
Листинг 1 CREATE TABLE Personnel (empCHAR(10)PRIMARY KEY, salaryDECIMAL(8,2)NOT NULL CHECK(salary > = 0.00), lftINTEGERNOT NULL, rgtINTEGERNOT NULL, CHECK(lft < rgt));
INSERT INTO Personnel VALUES ('Albert', 1000.00, 1, 28); INSERT INTO Personnel VALUES ('Bert', 900.00, 2, 5); INSERT INTO Personnel VALUES ('Charles', 900.00, 6, 19); INSERT INTO Personnel VALUES ('Diane', 900.00, 20, 27); INSERT INTO Personnel VALUES ('Edward', 750.00, 3, 4); INSERT INTO Personnel VALUES ('Fred', 800.00, 7, 16); INSERT INTO Personnel VALUES ('George', 750.00, 17, 18); INSERT INTO Personnel VALUES ('Heidi', 800.00, 21, 26); INSERT INTO Personnel VALUES ('Igor', 500.00, 8, 9); INSERT INTO Personnel VALUES ('Jim', 100.00, 10, 15); INSERT INTO Personnel VALUES ('Kathy', 100.00, 22, 23); INSERT INTO Personnel VALUES ('Larry', 100.00, 24, 25); INSERT INTO Personnel VALUES ('Mary', 100.00, 11, 12); INSERT INTO Personnel VALUES ('Ned', 100.00, 13, 14);
Удаление одиночного узла в середине дерева тяжелее чем удаление полных поддеревьев. Когда Вы удаляете узел в середине дерева, Вы должны решить, как заполнить отверстие. Имеются два пути. Первый метод к повышает одного из детей к позиции первоначального узла (предположим, что отец умирает, и самый старший сын занимает бизнес. Самый старший потомок всегда показывается как крайний левый дочерний узел под родителем. Второй метод для удаления одиночного узла в середине дерева состоит в том, чтобы подключить потомков к предку первоначального узла (можно сказать, что мамочка умирает, и дети приняты бабушкой). Меня спрашивают, почему я не показываю больше процедурного кода в примерах. Прямо сейчас, ANSI и ISO пробуют договориться о стандартном процедурном языке для триггеров и хранимых процедур называемом SQL/PSM. Однако, этот стандарт еще не утвержден, что означает, что я должен использовать или свой собственный псевдо-код или выбирать одного производителя. Я решил использовать пока Английские комментарии, но я буду начинать использовать SQL/PSM, когда он будет завершен. |