ELO-320 Arboles Binarios PDF
ELO-320 Arboles Binarios PDF
ELO-320 Arboles Binarios PDF
Arboles Binarios
8: Arboles 1
8-Árboles Binarios
8.1 Definiciones y tipos de datos
8.2 Cálculos de complejidad
8.3 Operaciones básicas y de recorrido
8.4 Operaciones de consulta
8.5 Operaciones de modificación
8: Arboles 2
Definiciones
8: Arboles 3
Definiciones II
❒ Árboles binarios
❍ Cada nodo puede tener un hijo izquierdo y/o un
hijo derecho.
❍ Un árbol binario está formado por un nodo raíz,
un subárbol izquierdo I y uno derecho D.
• Donde I y D son árboles binarios. Los subárboles se
suelen representar gráficamente como triángulos.
8: Arboles 5
Definiciones IV
8: Arboles 7
8-Árboles Binarios
8.1 Definiciones y tipos de datos
8.2 Cálculos de complejidad
8.3 Operaciones básicas y de recorrido
8.4 Operaciones de consulta
8.5 Operaciones de modificación
8: Arboles 8
Complejidad: árboles completos
❒ Deduciremos, de manera inductiva ❒ Árbol de nivel 1.
la altura de las hojas en función del Nodos = 3 = 2 2-1
número de nodos.
❒ El caso más simple de un árbol Altura = 2
completo tiene tres nodos, un nivel
y altura (A) de dos.
❒ Hemos modificado levemente la ❒ Árbol de nivel 2.
definición de altura, como el Nodos = 7 = 23-1
número de nodos que deben ser Altura = 3
revisados desde la raíz a las hojas,
ya que la complejidad de los
algoritmos dependerá de esta ❒ Árbol de nivel 3.
variable. 4
❍ A medida que se avanza en la
Nodos = 15 = 2 -1
trayectoria se descarta T(n/2) Altura = 4
nodos en cada avance.
❒ Un árbol perfectamente balanceado
que tiene n nodos internos tiene n+1 ❒ Árbol de n nodos:
hojas. n = 2A-1
A = log2(n+1) = O(log n)
8: Arboles 9
Complejidad: árboles con un nivel de
desbalance
❒ Se ilustran los tres casos de ❒ Árboles de nivel 2.
árboles, de nivel dos, con un nivel Nodos de 4 a 6
de desbalance, para n = 4, 5 y 6.
De 23-1 hasta 23-2
❒ En general para n nodos se tiene:
2A-1 ≤ n ≤ 2A-2 Altura = 3
A <= (1 + log2(n) ) para la primera
desigualdad y A >= log2 (n +2) para
la segunda.
❒ Cual es la complejidad de A?
❒ Se pueden encontrar constantes ❒ Árboles de nivel 3.
(i.e. c1,c2) que acoten, por arriba Nodos de 8 a 14
y por abajo a ambas funciones
para n > 10,079: De 24-1 hasta 24-2
1*log2 n <=log2 (n+2) <= A <= 1 + log2n Altura = 4
<= 1.3* log2n ❒ Árbol de nivel m.
A = Θ(log n) Nodos: 2A-1 ≤ n ≤ 2A-2
Altura = A
8: Arboles 10
Complejidad: árboles construidos en
forma aleatoria
❒ Para n nodos, con claves: 1, 2, 3, …, n, se pueden construir n!
árboles. Ya que existen n! permutaciones de n elementos.
❒ El orden de los elementos de la permutación, es el orden en que se
ingresan las claves a partir de un árbol vacío.
❒ Lo que se desea conocer es la altura An, definida como la altura
promedio de las búsquedas de las n claves y promediadas sobre los
n! árboles que se generan a partir de las n! permutaciones que se
pueden generar con la n claves diferentes.
❒ Si el orden de llegada de las claves que se insertan al árbol se
genera en forma aleatoria, la probabilidad de que la primera clave,
que es la raíz, tenga valor i es 1/n.
❒ Esto dado que todas las claves son
igualmente probables.
❒ El subárbol izquierdo contiene (i-1) nodos;
por lo tanto el subárbol derecho contiene (n-i) nodos.
❒ Para este tipo de árbol en promedio el largo
de cualquier trayectoria es: An ≈ 2ln(n) = O(log 2)
8: Arboles 11
Complejidad: árboles construidos en
forma aleatoria (cont)
❒ La gráfica muestra que para árboles de tipo 1000 nodos, deben
recorrerse cerca de nueve nodos desde la raíz hasta las hojas
(peor caso), si está balanceado.
❒ El largo promedio de los recorridos es 12 en un árbol generado
aleatoriamente, y 1000 en peor caso.
❒ Contrastando con un árbol
balanceado para n grande:
An/A≈ 2ln(n)/2log2(n)≈ 1,3
❒ Para n grande en promedio
el alargue es entre un
20 - 39%
8: Arboles 12
8-Árboles Binarios
8.1 Definiciones y tipos de datos
8.2 Cálculos de complejidad
8.3 Operaciones básicas y de recorrido
8.4 Operaciones de consulta
8.5 Operaciones de modificación
8: Arboles 13
Recorridos en árboles
8: Arboles 17
Operaciones básicas: recorrer II
❒ Para simplificar el cálculo podemos asumir un árbol balanceado:
T(n) = T(n/2) + Θ(1) + T(n/2 - 1)
❒ Y para grandes valores de n, podemos simplificar aún más:
T(n) = 2*T(n/2) que tiene por solución: T(n) = n = Θ(n)
8: Arboles 18
Operaciones básicas: recorrer mostrando
nivel
❒ Recorrer en orden mostrando nivel:
void inorder(pnodo t, int nivel)
{
if (t != NULL)
{
inorder(t->left, nivel+1);
printf ("%d %d \n", t->clave, nivel);
inorder(t->right, nivel +1);
}
}
❒ Ejemplo de uso:
inorder(arbol, 0); //Imprime considerando raíz de nivel 0.
❒ Mostrar en post-orden y pre-orden son análogos a como se
implemento RecorreEnOrden( ) e inorder( ).
8: Arboles 19
8-Árboles Binarios
8.1 Definiciones y tipos de datos
8.2 Cálculos de complejidad
8.3 Operaciones básicas y de recorrido
8.4 Operaciones de consulta
8.5 Operaciones de modificación
8: Arboles 20
Buscar mínimo
pnodo BuscarMinimoIterativo(pnodo t) {
while ( t != NULL)
{
if ( t->left == NULL )
return (t); //apunta al mínimo.
else
t=t->left; //desciende por la izquierda
}
return (t); /* NULL si árbol vacío*/
}
pnodo BuscaMinimoRec(pnodo t) {
if (t == NULL)
return(NULL); //si árbol vacío retorna NULL
else // Si no es vacío
if (t->left == NULL)
return(t); // Si no tiene hijo izquierdo: lo encontró.
else
return( BuscaMinimoRec (t->left) ); //busca en subárbol izquierdo.
} 8: Arboles 21
Buscar máximo
pnodo BuscarMaximoIterativo(pnodo t) {
while ( t != NULL) {
if ( t->right == NULL )
return (t); //apunta al máximo.
else
t=t->right; //desciende
}
return (t); /* NULL Si árbol vacío*/
}
pnodo BuscaMaximoRec(pnodo t) {
if (t == NULL)
return(NULL); //si árbol vacío retorna NULL
if (t->right == NULL)
return(t ); // Si no tiene hijo derecho: lo encontró.
return( BuscaMaximoRec (t->right) ); //sigue buscando en subárbol der.
}
8: Arboles 22
Nodo descendiente del subárbol derecho
con menor valor de clave
pnodo MenorDescendienteSD(pnodo t) {
if (t == NULL)
return(NULL); //si árbol vacío retorna NULL
if (t->right == NULL)
return(NULL ); // Si no tiene hijo derecho no hay sucesor.
return( BuscaMinimo (t->right) ); //sigue buscando en subárbol der.
}
8: Arboles 23
Nodo descendiente del subárbol derecho
con menor valor de clave II
❒ Para el diseño iterativo, deben estudiarse dos
casos:
❍ El caso D1, un nodo sin hijo izquierdo, indica que se
encontró el mínimo.
❍ El caso D2, debe descenderse por el subárbol
derecho de t, por la izquierda, mientras se tengan
hijos por la izquierda.
8: Arboles 24
Nodo descendiente del subárbol
derecho con menor valor de clave III
❒ Menor descendiente de subárbol derecho
pnodo MenorDescendienteIterativoSD(pnodo t) {
pnodo p;
if (t == NULL)
return(NULL); // si árbol vacío retorna NULL
if (t->right == NULL)
return(NULL ); // sin hijo derecho no hay sucesor.
else
p = t->right;
while ( p->left != NULL)
{ // mientras no tenga hijo izq descender por izq.
p = p->left;
} // al terminar while p apunta al menor descendiente
return (p); // retorna el menor
8: Arboles
} 25
Nodo sucesor
❒ Dado un nodo encontrar su sucesor no es el mismo
problema anterior, ya que el nodo podría ser una
hoja o un nodo sin subárbol derecho.
❒ Por ejemplo en la Figura, el sucesor del nodo con
clave 4 es el nodo con clave 5. El sucesor del nodo
2 es el nodo con valor 3.
❒ Se requiere disponer de un puntero al padre del
nodo, para que la operación sea de costo
logarítmico, en promedio.
❒ Si un nodo tiene subárbol derecho, el sucesor de
ese nodo es el ubicado más a la izquierda en ese
subárbol; si no tiene subárbol derecho, es el
menor ancestro (que está sobre el nodo en la
trayectoria hacia la raíz) que tiene a ese nodo
(e.g. 4) en su subárbol izquierdo.
❒ Como en peor caso debe ascenderse un
trayectoria del nodo hacia la raíz, el costo será
O(a), donde a es la altura del árbol. 8: Arboles 26
Nodo sucesor
❒ Algoritmo Sucesor:
Si el árbol no es vacío:
Si no tiene subárbol derecho:
Mientras exista el padre y éste apunte al nodo
dado por la derecha se asciende:
Hasta encontrar el primer padre por la
izquierda.
Si no existe ese padre, se retorna NULL, t
era el nodo con valor máximo
Si tiene subárbol derecho, el sucesor es el mínimo del
subárbol derecho. 8: Arboles 27
Nodo sucesor
pnodo Sucesor(pnodo t) {
pnodo p;
if (t == NULL)
return(NULL); //si árbol vacío retorna NULL
if (t->right == NULL) {
p = t->padre; //p apunta al padre de t
while ( p!=NULL && t == p->right) {
t=p; p=t->padre;
} //se asciende
return(p); //
}
else
return( BuscaMinimo (t->right) ); //busca mín. en subárbol der.
}
8: Arboles 28
Algoritmos relacionados
❒ Nodo descendiente del subárbol
izquierdo con mayor valor de
clave.
❍ Basta intercambiar left por right,
right por left y min por max en los
diseños desarrollado previamente
para MenorDescendienteSD.
❒ Predecesor.
❍ El código de la función predecesor es
la imagen especular del código de
sucesor.
8: Arboles 29
Buscar
❒ Debido a la propiedad de los árboles binarios de
búsqueda, si el valor buscado no es igual al de
nodo actual, sólo existen dos posibilidades: que
sea mayor o que sea menor.
Lo que implica que el nodo buscado puede
pertenecer a uno de los dos subárboles.
❒ Cada vez que se toma la decisión de buscar en
uno de los subárboles de un nodo, se están
descartando los nodos del otro subárbol.
❒ En caso de árboles balanceados, se descarta la
mitad de los elementos de la estructura, esto
cumple el modelo: T(n) = T(n/2) +c, lo cual
asegura complejidad logarítmica. 8: Arboles 30
Buscar
pnodo BuscarIterativo( pnodo t, int valor) {
while ( t != NULL) {
if ( t->clave == valor ) // se debe implementar para distintos
return (t); // tipos de datos
else {
if (t->clave < valor )
t = t->right; //desciende por la derecha
else
t = t->left; //desciende por la izquierda
}
}
return (t); /* NULL No lo encontró*/
}
8: Arboles 31
Buscar II
pnodo BuscarRecursivo( pnodo t, int valor ) {
if ( t == NULL)
return (NULL); // árbol vacío o hijo de hoja
else {
if ( t->clave == valor )
return(t); // lo encontró
else {
if ( t->clave > valor )
t = BuscarRecursivo ( t->left, valor);
else
t = BuscarRecursivo ( t->right, valor);
}
}
return ( t ); // los retornos de las llamadas recursivas se pasan via t
8: Arboles
} 32
Buscar III
❒ Pueden eliminarse las asignaciones y el retorno final de esta forma:
pnodo BuscarRecursivo2( pnodo t, int valor ) {
if ( t == NULL)
return (NULL); /* árbol vacío o hijo de hoja */
else {
if ( t->clave == valor )
return (t); /* lo encontró */
else {
if ( t->clave > valor )
return ( BuscarRecursivo2 ( t->left, valor) );
else
return ( BuscarRecursivo2 ( t->right, valor)) ;
}
}
} 8: Arboles 33
Buscar: Complejidad
❒ Si T(a) es la complejidad de la búsqueda en un
árbol de altura a. En cada iteración, el problema
se reduce a uno similar, pero con la altura
disminuida en uno, y tiene costo constante el
disminuir la altura.
❒ Entonces: T(a) = T(a-1) + Θ(1)
❒ La solución de esta recurrencia, es:
T(a) = a Θ(1) = Θ(a)
Pero en árboles de búsqueda se tiene que:
log n ≤ a ≤ n
Entonces:
Θ( log n) ≤ T(a) ≤ Θ(n) 8: Arboles 34
8-Árboles Binarios
8.1 Definiciones y tipos de datos
8.2 Cálculos de complejidad
8.3 Operaciones básicas y de recorrido
8.4 Operaciones de consulta
8.5 Operaciones de modificación
8: Arboles 35
Insertar Nodos: Iterativo I
❒ Primero se busca el sitio para
insertar. Si el valor que se desea
insertar ya estaba en el árbol, no
se efectúa la operación; ya que no
se aceptan claves duplicadas.
Entonces: se busca el valor; y si no
está, se inserta el nuevo nodo.
❒ Es preciso almacenar en una
variable local q, la posición de la
hoja en la que se insertará el nuevo
nodo. q permite conectar el nuevo
nodo creado al árbol.
❒ Se recorre una trayectoria de la
raíz hasta una hoja para insertar.
Entonces, si a es la altura, la
complejidad de la inserción es:
T(a). 8: Arboles 36
Insertar Nodos: Iterativo I
typedef enum {left, right, vacio} modo;
/*Al salir del while q apunta al nodo en
pnodo InsertarIterativo(pnodo t, int valor) { el arbol donde se insertará el nuevo
pnodo q= t; modo porlado=vacio; nodo, y porlado la dirección */
while ( t != NULL) { /* El argumento t apunta a NULL */
if ( t->clave == valor ) {
t = CreaNodo(valor);
/* lo encontró, no inserta nodo */
return (t); if (porlado==left)
} q->left=t;
else { else if(porlado==right)
q=t ;
q->right=t;
if (t->clave < valor){
return (t); /* Apunta al recién
t = t->right;
insertado. Null si no se pudo
porlado=right;
insertar*/
}
}
else {
t = t->left;
porlado=left;
}
} 8: Arboles 37
}
Insertar Nodos: Iterativo II
pnodo Insertar(pnodo t, int valor) int main()
{ {
pnodo *p = &t; pnodo pRoot = NULL;
while (*p != NULL) pRoot = Insertar(pRoot, 5);
{ Insertar(pRoot, 3);
if ((*p)->clave < valor) Insertar(pRoot, 7);
p = &((*p)->right); Insertar(pRoot, 1);
else if ((*p)->clave > valor) Insertar(pRoot, 4);
p = &((*p)->left);
else return 0;
{ }
/* Ya estaba. No hace nada */
return t;
}
}
*p = CreaNodo(valor);
return t;
}
8: Arboles 38
Insertar Nodos: Recursivo
pnodo InsertarRecursivo( pnodo t, int valor) {
if (t == NULL) {
t = CreaNodo(valor); //insertar en árbol vacío o en hoja.
}
else if (valor < t->clave) { //insertar en subárbol izquierdo.
t->left = InsertarRecursivo(t->left, valor);
}
else if (valor > t->clave) {//insertar en subárbol derecho
t->right = InsertarRecursivo (t->right, valor);
}
/* else: valor ya estaba en el árbol. No hace nada. */
return(t);
}
8: Arboles 39
Descartar Nodos
❒ Primero se busca el nodo cuyo valor de
clave es igual al valor pasado como
argumento. Si no lo encuentra retorna
NULL. Si lo encuentra, se producen varios
casos. Lo importante es mantener la
vinculación entre los elementos del árbol:
❍ a) El nodo que se desea descartar es una
hoja. En este caso, la operación es trivial,
basta escribir un puntero con valor nulo. La
estructura se conserva.
❍ b) El nodo que se desea descartar es un
nodo interno. i) con un hijo por la izquierda
o la derecha el padre debe apuntar al nieto,
para conservar la estructura de árbol. Esto
implica mantener un puntero al padre, en el
descenso.
8: Arboles 40
Descartar Nodos
❒ b) El nodo que se desea descartar
es un nodo interno.
i) con un hijo por la izquierda o la
derecha el padre debe apuntar al
nieto, para conservar la estructura
de árbol. Esto implica mantener un
puntero al padre, en el descenso.
ii) con dos hijos para conservar la
estructura del árbol, se debe
buscar I, el mayor descendiente
del hijo izquierdo; o bien D, el
menor descendiente del hijo
derecho. Luego reemplazar la hoja
obtenida por el nodo a descartar.
❍ Se muestra la operación buscando D.
8: Arboles 41
Descartar Nodos II
pnodo Descartar(pnodo t, int valor) {
pnodo temp;
if (t == NULL)
printf("Elemento no encontrado\n");
else if (valor < t->clave) /* por la izquierda */
t->left = Descartar(t->left, valor);
else if (valor > t->clave) /* por la derecha */
t->right = Descartar(t->right, valor);
else { /* se encontró el elemento a descartar */
if (t->left && t->right) { /* dos hijos: remplazar con D */
temp = MenorDescendiente(t->right) ;
t->clave = temp->clave; //copia el nodo y borra la hoja
t->right = Descartar(t->right, temp->clave);
}
else { /* un hijo o ninguno */
...continua...
8: Arboles 42
Descartar Nodos III
...continuacion...
pnodo deltree(pnodo t) {
if (t != NULL) {
t->left = deltree(t->left);
t->right = deltree(t->right);
free(t);
}
return NULL;
}
8: Arboles 44
Profundidad del Árbol
int Profundidad(pnodo t) {
int left=0, right = 0;
if(t==NULL)
return 0; //Si árbol vacío, profundidad 0
if(t->left != NULL)
left = Profundidad(t->left); //calcula prof. sub arb. I
if(t->right != NULL)
right = Profundidad(t->right); //calcula prof. sub arb.D
if( left > right) //si el izq tiene mayor profundidad
return left+1; //retorna profundidad del sub arb izq + 1
else
return right+1; //retorna prof. del sub arb der + 1
}
8: Arboles 45
Altura del Árbol
int Altura(pnodo T) {
int h, max;
if (T == NULL)
return -1;
else {
h = Altura (T->left);
max = Altura (T->right);
if (h > max)
max = h;
return(max+1);
}
}
8: Arboles 46
Numero de Hojas del Árbol
int NumerodeHojas(pnodo t) {
int total = 0; //Si árbol vacío, no hay hojas
if(t==NULL)
return 0;
// Si es hoja, la cuenta
if(t->left == NULL && t->right == NULL)
return 1;
//cuenta las hojas del subárbol izquierdo
if(t->left!= NULL)
total += NumerodeHojas(t->left);
//cuenta las hojas del subárbol derecho
if(t->right!=0)
total += NumerodeHojas(t->right);
return total; //total de hojas en subárbol
} 8: Arboles 47
Contar Nodos del Arbol
int ContarNodos(pnodo t) {
if (t == NULL)
return 0;
return (1 + ContarNodos(t->left) +
ContarNodos(t->right) );
}
8: Arboles 48
Partir el Arbol
pnodo split(int key, pnodo t,
pnodo *l, pnodo *r) {
while (t != NULL && t- if (t == NULL) {
>clave != key) { *l = NULL;
if (t->clave < key) { *r = NULL;
*l = t; }
t = t->right; else { /* t->clave == key */
l = &((*l)->right); *l = t->left;
*r = t->right;
} else {
}
*r = t;
return t;
t = t->left;
}
r = &((*r)->left);
}
} // fin del while
8: Arboles 49
Insertar Nueva Raíz
pnodo InsertarRaiz(int key, pnodo t) {
pnodo l, r;
t = split(key, t, &l, &r);
if (t == NULL) {
t = CreaNodo(key);
t->left = l;
t->right = r;
}
else {
t->left = l;
t->right = r;
Error();
}
return t;
}
8: Arboles 50
Unir dos Árboles
pnodo join(pnodo l, pnodo r) {
pnodo t = NULL; if (l == NULL)
pnodo *p = &t; *p = r;
while (l != NULL && r != NULL) { else /* (r == NULL)
if (rand()%2) { //cara y sello.
*/
*p = l;
p = &((*p)->right);
*p = l;
l = l->right; return t;
} else { }
*p = r;
p = &((*p)->left);
r = r->left;
}
} // fin del while
8: Arboles 51