Решение Таблица замыканий — простой и элегантный способ хранения иерархий. Он реализует хранение всех путей по дереву, а не только путей с прямыми связями «родитель-потомок».
Помимо простой таблицы Comments создайте еще одну таблицу TreePaths с двумя столбцами, каждый из которых является внешним ключом для таблицы Comments.
Файл npvmepa:_Trees/soln/closure-table/create-table.sql
CREATE TABLE Comments (
comment_id SERIAL PRIMARY KEY,
bug_id BIGINT UNSIGNED NOT NULL, author BIGINT UNSIGNED NOT NULL, comment_date DATETIME NOT NULL, comment TEXT NOT NULL,
FOREIGN KEY (bug_id) REFERENCES Bugs(bug_id), FOREIGN KEY (author) REFERENCES Accounts(account_id)) ;
CREATE TABLE TreePaths (
ancestor BIGINT UNSIGNED NOT NULL,
descendant BIGINT UNSIGNED NOT NULL,
PRIMARY KEY (ancestor, descendant),
FOREIGN KEY (ancestor) REFERENCES Comments(comment_id), FOREIGN KEY (descendant) REFERENCES Comments(comment_id)) ;
Вместо таблицы Comments для хранения информации о структуре дерева используйте таблицу TreePaths. Храните здесь одну строку для каждой пары узлов, связанных отношением «предок/потомок», даже если они разделяются в дереве несколькими уровнями. Также добавьте строку для каждого узла, чтобы ссылаться на него. Иллюстрация связывания узлов в пары показана на рис. 3.4.
предок потомок предок потомок предок потомок
I 1 1 7 4 6
1 2 2 2 4 7
1 3 2 3 5 5
1 4 3 3 6 6
1 5 4 4 6 7
1 6 4 5 7 7
Запросы для извлечения предков и потомков из данной таблицы становятся более простыми, чем запросы в решении Вложенные множества. Чтобы извлечь потомков комментария № 4, сравните строки в TreePaths, где ancestor — 4:
Файл примера: Trees/soln/closure-table/descendants.sql
SELECT с*
FROM Comments AS с
JOIN TreePaths AS t ON с.comment_id = t.descendant WHERE t.ancestor = 4;
Чтобы извлечь предков комментария № 6, подберите строки в TreePaths, где потомок — 6:
Файл прчмьра: Trees/soln/closure-table/ancestors.sql
SELECT с*
FROM Comments AS с
JOIN TreePaths AS t ON c.comment_id = t.ancestor
WHERE t.descendant = 6;
Чтобы вставить новый лист, например дочерний элемент для комментария № 5, сначала вставьте строку, ссылающуюся на саму себя. Затем добавьте копию набора строк в TreePaths, которые ссылаются на комментарий № 5 как на descendant (включая строку, где комментарий № 5 ссылается на самого себя), заменив descendant на номер нового комментария:
Файлпримера:. Trees/soln/closure - table/insert.sql
INSERT INTO TreePaths (ancestor, descendant) SELECT t.ancestor, 8 FROM TreePaths AS t WHERE t.descendant = 5
UNION ALL
SELECT 8, 8;
Чтобы удалить концевую вершину, например комментарий № 7, удалите все строки в TreePaths, которые ссылаются на комментарий № 7 как на потомка (descendant):
Файл t\ptoMepa:_Trees/soln/closure-table/delete-leaf.sql
DELETE FROM TreePaths WHERE descendant = 7;
Чтобы удалить поддерево полностью, например комментарий № 4 и его потомков, удалите все строки в TreePaths, которые ссылаются на комментарий № 4 как на потомка (descendant), а также все строки, которые ссылаются на любых потомков комментария № 4 как на потомков:
Файл примера:.Trees/soln/closure-table/delete-subtree.sql
DELETE FROM TreePaths
WHERE descendant IN (SELECT descendant FROM TreePaths WHERE ancestor = 4) ;
Опубликовал vovan666
June 18 2013 12:25:12 ·
0 Комментариев ·
6494 Прочтений ·
• Не нашли ответ на свой вопрос? Тогда задайте вопрос в комментариях или на форуме! •
Комментарии
Нет комментариев.
Добавить комментарий
Рейтинги
Рейтинг доступен только для пользователей.
Пожалуйста, залогиньтесь или зарегистрируйтесь для голосования.
Нет данных для оценки.
Гость
Вы не зарегистрированны? Нажмите здесь для регистрации.