Pregunta

Tengo un árbol / antepasado / consulta problema no soy capaz de resolver:

Tengo una tabla con datos de menú y una tabla que contiene todos los antepasados del menú:

table menu               table ancestors
+-----+------------+--------+     +---------+--------------+-------+
| id  |      title | active |     | menu_id |  ancestor_id | level |
+-----+------------+--------+     +---------+--------------+-------+
|   1 |       Home |      0 |     |       1 |            0 |     0 |
|   2 |       News |      0 |     |       2 |            1 |     1 |
|   3 |        Foo |      0 |     |       3 |            2 |     2 |
|   4 |        Bar |      1 |     |       3 |            1 |     1 |
|   5 |  Downloads |      1 |     |       4 |            3 |     3 |
+-----+------------+--------+     |       4 |            2 |     2 |
                                  |       4 |            1 |     1 |
                                  |       5 |            1 |     1 |
                                  +---------+--------------+-------+

Puedo obtener todos los activos de las entradas de menú con sus antepasados fácilmente con:

 SELECT menu.id, menu.title, GROUP_CONCAT(ancestors.ancestor_id) as ancestors
FROM menu, ancestors
WHERE menu.active = 1
GROUP BY (menu.id);

 +----+-----------+----------+
 | id |     title |ancestors |
 +----+-----------+----------+
 |  4 |       Bar | 3,2,1    | 
 |  5 | Downloads | 1        |
 +----+-----------+----------+

Pero ¿cómo puedo obtener todo el para el árbol, es necesario antepasados, también?En mi resultado me haría falta la entrada de Foo y Noticias, así que tengo un consistente árbol.Se debe tener este aspecto:

 +----+-----------+----------+
 | id |     title |ancestors |
 +----+-----------+----------+
 |  2 |      News | 1        | 
 |  3 |       Foo | 2,1      | 
 |  4 |       Bar | 3,2,1    | 
 |  5 | Downloads | 1        |
 +----+-----------+----------+

¿Cómo ha sido la consulta a ser?

¿Fue útil?

Solución

Cuando hago esto, estructuro la tabla ancestors de forma ligeramente diferente. En lugar de level, almaceno pathlength. También almacenar una fila para cada elemento de menú para apuntar a sí mismo, con una longitud de recorrido de 0.

+---------+--------------+------------+
| menu_id |  ancestor_id | pathlength |
+---------+--------------+------------+
|       1 |            1 |          0 |
|       2 |            2 |          0 |
|       3 |            3 |          0 |
|       4 |            4 |          0 |
|       5 |            5 |          0 |
|       2 |            1 |          1 |
|       3 |            2 |          2 |
|       3 |            1 |          1 |
|       4 |            3 |          3 |
|       4 |            2 |          2 |
|       4 |            1 |          1 |
|       5 |            1 |          1 |
+---------+--------------+------------+

Estas entradas "reflexivos" permiten a unirse al conjunto de elementos de menú activos a la mesa de cierre. Cambiando el nivel de longitud de trayectoria le permite excluir las entradas reflexivas del conjunto de los antepasados.

Ahora se puede consultar todos los elementos del menú que son ancestros de los elementos "activos" del menú, incluyendo los elementos de menú activos propios:

SELECT a2.menu_id, m2.title, GROUP_CONCAT(a2.ancestor_id) AS ancestors
FROM menu m1
JOIN ancestors a1 ON (m1.id = a1.menu_id)
JOIN ancestors a2 ON (a1.ancestor_id = a2.menu_id AND a2.pathlength > 0)
JOIN menu m2 ON (a2.menu_id = m2.id)
WHERE m1.active = 1
GROUP BY a2.menu_id;

Resultados:

+---------+-----------+-----------+
| menu_id | title     | ancestors |
+---------+-----------+-----------+
|       2 | News      | 1         | 
|       3 | Foo       | 2,1       | 
|       4 | Bar       | 3,2,1     | 
|       5 | Downloads | 1         | 
+---------+-----------+-----------+
Licenciado bajo: CC-BY-SA con atribución
No afiliado a StackOverflow
scroll top