Árvore binária
Observação: esta aula será substituída depois pelo tutorial completo. No momento, devido a restrições do IF SE não consegui fazer o upload do tutorial.
Uma árvore binária é uma estrutura de dados caracterizada por:
- Ou não tem elemento algum (árvore vazia).
- Ou tem um elemento distinto, denominado raiz, com dois ponteiros para duas estruturas diferentes, denominadas sub-árvore esquerda e sub-árvore direita.
Definições para árvores binárias
Os nós de uma árvore binária possuem graus zero, um ou dois. Um nó de grau zero é denominado folha.Em uma árvore binária, por definição, cada nó poderá ter até duas folhas, sendo que ela se compara com a abb (árvore binária de busca), apesar de não ter a propriedade da mesma ("na abb, existe uma regra na inserção").
A profundidade de um nó é a distância deste nó até a raiz. Um conjunto de nós com a mesma profundidade é denominado nível da árvore. A maior profundidade de um nó, é a altura da árvore.
Uma árvore "estritamente binária" é uma árvore na qual todo nó tem zero ou duas folhas.
Existem autores, porém, que adotam essa definição para o termo quase completa, e utilizam o termo completa apenas para árvores em que todos os níveis têm o máximo número de elementos.
Definições em teoria dos grafos
Em teoria dos grafos, uma árvore binária é definida como um grafo acíclico, conexo, dirigido e que cada nó não tem grau maior que 2. Assim sendo, só existe um caminho entre dois nós distintos.E cada ramo da árvore é um vértice dirigido, sem peso, que parte do pai e vai o filho.
Inserção
O algoritmo da inserção em árvores binárias são facilmente implementadas através de função recursiva.Algoritmo da inserção em C:
void inserir(struct No **pRaiz, int numero){ if(*pRaiz == NULL){ * pRaiz = (struct No *) malloc(sizeof(struct No)); (*pRaiz)→pEsquerda = NULL; (*pRaiz)→pDireira = NULL; (*pRaiz)→numero = numero; }else{ if(numero <(*pRaiz)→numero) inserir(&(*pRaiz)→pEsquerda, numero)); else inserir(&(*pRaiz)→pDireita, numero)); } }
Percursos em árvore
Existem três tipos de percursos: Percurso em ordem simétrica(em-ordem), pré-ordem e pós-ordem.Ordem Simétrica
O algoritmo recursivo desse percurso em C é:void emOrdem(struct No *pNo) { if(pNo != NULL) { emOrdem(pNo→pEsquerda); visita(pNo); emOrdem(pNo→pDireita); } }
Para a árvore acima, o percurso seria: 2, 7, 5, 6, 11, 2, 5, 4 e 9. Veja a linha de execução:
- emOrdem(2→esq)
- emOrdem(7→esq)
- emOrdem(2→esq)
- visita(2)
- emOrdem(2→dir)
- visita(7)
- emOrdem(7→dir)
- emOrdem(6→esq)
- emOrdem(5→esq)
- visita(5)
- emOrdem(5→dir)
- visita(6)
- emOrdem(6→dir)
- emOrdem(11→esq)
- visita(11)
- emOrdem(11→dir)
- emOrdem(6→esq)
- emOrdem(7→esq)
- visita(2)
- emOrdem(2→dir)
- emOrdem(5→esq)
- visita(5)
- emOrdem(5→dir)
- emOrdem(9→esq)
- emOrdem(4→esq)
- visita(4)
- emOrdem(4→dir)
- visita(9)
- emOrdem(9→dir)
- emOrdem(9→esq)
Pré-Ordem
O algoritmo recursivo desse percurso em C é:void preOrdem(Struct No *pNo){ if(pNo != NULL){ visita(pNo); preOrdem(pNo→pEsquerda); preOrdem(pNo→pDireita); } }
Para a árvore acima, o percurso seria: 2, 7, 2, 6, 5, 11, 5, 9 e 4. Veja a linha de execução:
- visita(2)
- preOrdem(2→esq)
- visita(7)
- preOrdem(7→esq)
- visita(2)
- preOrdem(2→esq)
- preOrdem(2→dir)
- preOrdem(7→dir)
- visita(6)
- preOrdem(6→esq)
- visita(5)
- preOrdem(5→esq)
- preOrdem(5→dir)
- preOrdem(6→dir)
- visita(11)
- preOrdem(11→esq)
- preOrdem(11→dir)
- preOrdem(2→dir)
- visita(5)
- preOrdem(5→esq)
- preOrdem(5→dir)
- visita(9)
- preOrdem(9→esq)
- visita(4)
- preOrdem(4→esq)
- preOrdem(4→dir)
- preOrdem(9→dir)
Pós-Ordem
O algoritmo recursivo desse percurso em C é:void posOrdem(Struct No *pNo){ if(pNo != NULL){ posOrdem(pNo→pEsquerda); posOrdem(pNo→pDireita); visita(pNo); } }
Para a árvore acima, o percurso seria: 2, 5, 11, 6, 7, 4, 9, 5 e 2. Veja a linha de execução:
- posOrdem(2→esq)
- posOrdem(7→esq)
- posOrdem(2→esq)
- posOrdem(2→dir)
- visita(2)
- posOrdem(7→dir)
- posOrdem(6→esq)
- posOrdem(5→esq)
- posOrdem(5→dir)
- visita(5)
- posOrdem(6→dir)
- posOrdem(11→esq)
- posOrdem(11→dir)
- visita(11)
- visita(6)
- posOrdem(6→esq)
- visita(7)
- posOrdem(7→esq)
- posOrdem(2→dir)
- posOrdem(5→esq)
- posOrdem(5→dir)
- posOrdem(9→esq)
- posOrdem(4→esq)
- posOrdem(4→dir)
- visita(4)
- posOrdem(9→dir)
- visita(9)
- posOrdem(9→esq)
- visita(5)
- visita(2)
Soma Elementos
public int SomaElementos(No raiz){ if (raiz != null){ soma=this.SomaElementos(raiz.p_Esq)+this.SomaElementos(raiz.p_Dir)+raiz.ObtemValor(); return soma; }else return 0; }
Maior Elemento
public int MaiorElemento(No raiz){ if (raiz != null){ if (maior < raiz.ObtemValor()) maior = raiz.ObtemValor(); this.MaiorElemento(raiz.p_Esq); this.MaiorElemento(raiz.p_Dir); return maior; }else return 0; }
Menor Elemento
public int MenorElemento(No raiz){ if (raiz != null){ if (menor > raiz.ObtemValor()) menor = raiz.ObtemValor(); this.MenorElemento(raiz.p_Esq); this.MenorElemento(raiz.p_Dir); return menor; }else return 0; }
Transformação de uma árvore n-ária
Uma árvore n-ária qualquer (árvore cujos nós possuem graus menores ou iguais a n) podem ser representados por uma árvore binária. Um exemplo dessa transformação pode ser vista emMétodos para representação de árvores binárias
Uma das maneiras mais simples de representar árvores binárias em linguagens de programação é por meio de arranjos unidimensionais (vetores). Caso a raiz esteja na posição zero, dado um nó de índice i qualquer, os seus filhos terão índices e e o seu pai terá índice piso((i - 1)/2). Caso a raiz esteja na posição um, os filhos terão índices e e o pai terá índice piso(i/2). Essa implementação é utilizada para representar árvores completas ou quase completas. Heaps, que são árvores binárias quase completas são implementadas na forma de um vetor de uma maneira bastante eficiente.Apesar da simplicidade, essa representação requer uma grande quantidade de memória contígua para o armazenamento de árvores grandes, o crescimento desta é díficil de implementar e manter e pode haver grande quantidade de desperdício de memória.
Em uma linguagem que possua suporte a estruturas e referências (por exemplo pascal e C), as árvores podem ser implementadas a partir de nós, com um, ou mais, campos para a(s) informação(ões) principal(is) e dois campos apontadores especiais, denominados esquerda e direita, que fazem referência às sub-árvores esquerda e direita, respectivamente. Algumas vezes, há um apontador para o pai. Em um nó do tipo folha, os campos apontadores possuem valores especiais (nil em Pascal e "NULL" em C).
Árvore binária de busca
- Nós - são todos os itens guardados na árvore
- Raiz - é o nó do topo da árvore (no caso da figura acima, a raiz é o nó 8)
- Filhos - são os nós que vem depois dos outros nós (no caso da figura acima, o nó 6 é filho do 3)
- Pais - são os nós que vem antes dos outros nós (no caso da figura acima, o nó 10 é pai do 14)
- Folhas - são os nós que não têm filhos; são os últimos nós da árvore (no caso da figura acima, as folhas são 1, 4, 7 e 13)
Operações
Operações em uma árvore binária requerem comparações entre nós. Essas comparações são feitas com chamadas a um comparador, que é uma subrotina que calcula a ordem total (ordem linear) em dois valores quaisquer. Esse comparador pode ser explícita ou implicitamente definido, dependendo da linguagem em que a árvore binária de busca está implementada.Busca
A busca em uma árvore binária por um valor específico pode ser um processo recursivo ou iterativo. Essa explicação usará um método recursivo.A busca começa examinando o nó raiz. Se a árvore está vazia, o valor procurado não pode existir na árvore. Caso contrário, se o valor é igual a raiz, a busca foi bem sucedida. Se o valor é menor do que a raiz, a busca segue pela subárvore esquerda. Similarmente, se o valor é maior do que a raiz, a busca segue pela subárvore direita. Esse processo é repetido até o valor ser encontrado ou a subárvore ser nula (vazia). Se o valor não for encontrado até a busca chegar na subárvore nula, então o valor não deve estar presente na árvore.
Inserção
A inserção começa com uma busca, procurando pelo valor, mas se não for encontrado, procuram-se as subárvores da esquerda ou direita, como na busca. Eventualmente, alcança-se a folha, inserindo-se então o valor nesta posição. Ou seja, a raiz é examinada e introduz-se um nó novo na subárvore da esquerda se o valor novo for menor do que a raiz, ou na subárvore da direita se o valor novo for maior do que a raiz. Abaixo, um algoritmo de inserção em Python:def arvore_binaria_inserir(no, chave, valor): if no is None: return TreeNode(None, chave, valor, None) if chave == no.chave: return TreeNode(no.filho_esquerdo, chave, valor, no.filho_direito) if chave < no.chave: return TreeNode(arvore_binaria_inserir(no.filho_esquerdo, chave, valor), no.chave, no.valor, no.filho_direito) else: return TreeNode(no.filho_esquerdo, no.chave, no.valor, arvore_binaria_inserir(no.filho_direito, chave, valor))
Exclusão
A exclusão de um nó é um processo mais complexo. Para excluir um nó de uma árvore binária de busca, há de se considerar três casos distintos para a exclusão:Exclusão na folha
A exclusão na folha é a mais simples, basta removê-lo da árvore.Exclusão de um nó com um filho
Excluindo-o, o filho sobe para a posição do pai.Exclusão de um nó com dois filhos
Neste caso, pode-se operar de duas maneiras diferentes. Pode-se substituir o valor do nó a ser retirado pelo valor sucessor (o nó mais à esquerda da subárvore direita) ou pelo valor antecessor (o nó mais à direita da subárvore esquerda), removendo-se aí o nó sucessor (ou antecessor).No exemplo acima, o nó de valor 30 está para ser removido, e possui como sucessor imediato o valor 35 (nó mais à esquerda da sua sub-árvore direita). Assim sendo, na exclusão, o valor 35 será promovido no lugar do nó a ser excluído, enquanto a sua sub-árvore (direita) será promovida para sub-árvore esquerda do 40, como pode ser visto na figura. Ao excluir, ou mesmo ao incluir, um nó, pode haver o desbalanceamento da árvore, que pode ser corrigido, por exemplo, com o balanceamento AVL.
Embora esta operação não percorra sempre a árvore até uma folha, esta é sempre uma possibilidade; assim, no pior caso, requer o tempo proporcional à altura da árvore, visitando-se cada nó somente uma única vez.
Transversal
Em uma árvore binária de busca podem-se fazer os três percursos que se fazem para qualquer árvore binária (percursos em inordem, pré-ordem e pós-ordem). É interessante notar que, quando se faz um percurso em ordem em uma árvore binária de busca, os valores dos nós aparecem em ordem crescente. A operação "Percorre" tem como objetivo percorrer a árvore numa dada ordem, enumerando os seus nós. Quando um nó é enumerado, diz-se que ele foi "visitado".Pré-ordem (ou profundidade):
- Visita a raiz
- Percorre a subárvore esquerda em pré-ordem
- Percorre a subárvore direita em pré-ordem
- Percorre a subárvore esquerda em ordem simétrica
- Visita a raiz
- Percorre a subárvore direita em ordem simétrica
- Percorre a subárvore esquerda em pós-ordem
- Percorre a subárvore direita em pós-ordem
- Visita a raiz
def arvore_binaria_transversal(nó_arvore): if nó_arvore is None: return esquerda, nó_valor, direita = nó_arvore arvore_binaria_transversal(esquerda) visite(nó_valor) arvore_binaria_transversal(direita)
Ordenação
Uma árvore binária de busca pode ser usada para executar um simples algoritmo de ordenação. Para fazer isto, são introduzidos todos os valores desejados, classificado-se depois em uma árvore binária de busca, atravessando-a em ordem, construindo um novo resultado:def criar_arvore_binaria(valor): arvore = None for v in valor: arvore = arvore_binaria_de_insercao(arvore, v) return arvore def arvore_binaria_transversa(nó_arvore): if nó_arvore is None: return [] else: esquerda, valor, direita = nó_arvore return (arvore_binaria_transversal(esquerda) + [valor] + arvore_binaria_transversal(direita))
Nenhum comentário:
Postar um comentário
Observação: somente um membro deste blog pode postar um comentário.