Ir para conteúdo
Fórum Script Brasil
  • 0

Árvore Binária Completa


R. Bighi

Pergunta

Ae galera, estou tentando implementar um método que utiliza uma árvore binária (quase) completa para realizar suas operações. No caso, a inserção sempre vai da esquerda para a direita, e só o último nível é que pode ser incompleto, tendo somente os elementos mais à esquerda. Eu entendi o conceito, mas não consigo construir um algoritmo de inserção para essa árvore. Imagino que, a cada inserção de um nó, tenho que mover o ponteiro principal de volta à raiz, mas não estou conseguindo fazer funcionar porque não sei como fazer ele identificar onde deve inserir os próximos elementos para que sejam inseridos na ordem certa. Alguém pode me dar uma idéia??

Link para o comentário
Compartilhar em outros sites

3 respostass a esta questão

Posts Recomendados

  • 0

Então, o uso da árvore é para o seguinte: Suponha que eu tenha uma tabela hash com alguns valores inseridos. O método que estou tentando implementar vai utilizar a árvore pra encontrar uma possível alocação pra um novo valor nessa tabela, de forma que a busca nessa tabela seja a mais eficiente possível. Ainda não consegui completar o raciocinio, e por conta disso n tenho mt coisa da arvore pronta. Eu fiz as funções para a árvore e, meu problema tá sendo no main. Pensei em usar um while(*p->numero != 0) para continuar completando a árvore até encontrar uma posição da tabela que esteja vaga, MAS eu não consigo pensar em como percorrer cada nó da árvore na ordem certa pra usar a função de inserir. Entendeu? As funções que fiz são essas:


#include <stdio.h>
#include <stdlib.h>

typedef struct No{
int numero;
struct No *pEsquerda; //ponteiro apontando para o filho esquerdo
struct No *pDireita; //ponteiro apontando para o filho direito
struct No *p; //ponteiro para percorrer a arvore
}No;

void criarArvore(No **pRaiz){
*pRaiz = NULL; //inicializa a arvore
}

void inserir (No **pRaiz)
{
if (*p->pEsquerda == NULL) // Se o lado esquerdo da árvore estiver vazio, inserir
{
*p->Esqueda = hash(i) + inc(i);
}
else {
if (*p->pDireita == NULL) // Se o lado esquerdo estiver ocupado, adicionar no direito.
{
*p->pDireita = hash(i) + inc(tabela[i]);
}
}
}
[/codebox]

Queria mesmo era uma idéia de como fzer... Realmente empaquei em como percorrer isso aí... Obrigado desde já!

Link para o comentário
Compartilhar em outros sites

  • 0
Então, o uso da árvore é para o seguinte: Suponha que eu tenha uma tabela hash com alguns valores inseridos. O método que estou tentando implementar vai utilizar a árvore pra encontrar uma possível alocação pra um novo valor nessa tabela, de forma que a busca nessa tabela seja a mais eficiente possível. Ainda não consegui completar o raciocinio, e por conta disso n tenho mt coisa da arvore pronta. Eu fiz as funções para a árvore e, meu problema tá sendo no main. Pensei em usar um while(*p->numero != 0) para continuar completando a árvore até encontrar uma posição da tabela que esteja vaga, MAS eu não consigo pensar em como percorrer cada nó da árvore na ordem certa pra usar a função de inserir. Entendeu? As funções que fiz são essas:


#include <stdio.h>
#include <stdlib.h>

typedef struct No{
int numero;
struct No *pEsquerda; //ponteiro apontando para o filho esquerdo
struct No *pDireita; //ponteiro apontando para o filho direito
struct No *p; //ponteiro para percorrer a arvore
}No;

void criarArvore(No **pRaiz){
*pRaiz = NULL; //inicializa a arvore
}

void inserir (No **pRaiz)
{
if (*p->pEsquerda == NULL) // Se o lado esquerdo da árvore estiver vazio, inserir
{
*p->Esqueda = hash(i) + inc(i);
}
else {
if (*p->pDireita == NULL) // Se o lado esquerdo estiver ocupado, adicionar no direito.
{
*p->pDireita = hash(i) + inc(tabela[i]);
}
}
}
[/codebox]

Queria mesmo era uma idéia de como fzer... Realmente empaquei em como percorrer isso aí... Obrigado desde já!

Tem uns 15 dias que eu tive que fazer um trabalho para faculdade sobre Arvore

que pega alguns numero, faz o embaralhamento deste numero e ai insere na arvore assim que e inicializado o programa

O codigo tem as seguintes funcoes

printf("1 -> Inserir numero\n");//Insere novo numero

printf("2 -> Imprime ordem crescente\n");

printf("3 -> Imprime ordem decrescente\n");

printf("4 -> Imprime maior numero\n");

printf("5 -> Imprime menor numero\n");

printf("6 -> Imprime nos folhas\n");

printf("7 -> Imprime sub-arvore direita de um no\n");

printf("8 -> Imprime sub-arvore esquerda de um no\n");

printf("9 -> Imprime arvore\n");//imprime na tela no formato de arvore

printf("0 -> Finalizar o programa\n\n");

Codigo

[code]#include <stdio.h> #include <stdlib.h> #define max 10 typedef long TipoChave; typedef struct Registro {   TipoChave Chave;   /* outros componentes */ } Registro; typedef struct No * Apontador; typedef struct No {   Registro Reg;   Apontador Esq, Dir; } No; typedef Apontador TipoDicionario; //chamadas de funcoes void Inicializa(Apontador *p); void Permut( TipoChave A[], int n); void Insere(Registro x, Apontador *p); void TestaI(No *p, int pai); void Testa(No *p); void Cabecalho(); void OrdemCrescente(No *p); void OrdemDecrescente(No *p); void ImprimeArvore(int camada, No *p); void MaiorNumero(No *p); void MenorNumero(No *p); void NoFolhas(No *p); void ProcuraNumeroDir(No *p, int numero); void ProcuraNumeroEsq(No *p, int numero); int MenuOpcao(int opcao); double rand0a1(); int main(int argc, char *argv[]){          int opcao,         //variavel para indicar as opcoes p/ usuario         i,             //contador         numero;        //recebe o numero que o usuario quer inserir na arvore     No *Dicionario;     Registro x;     TipoChave vetor[max];          //inicializa o dicionario com NULL     Inicializa(&Dicionario);          //cria um vetor ordenado     for (i = 0; i < max; i++)     vetor[i] = i+1;          //embaralha os numeros no vetor     Permut(vetor,max-1);          //insere cada chave na arvore e testa sua integridade apos cada insercao     for (i = 0; i < max; i++){         x.Chave = vetor[i];         Insere(x, &Dicionario);         Testa(Dicionario);     }          while(opcao!=0){ // enquanto o usuario não digitar 0 o programa não ira finalizar              //funcao onde mostra o Menu de opcoes para usuario         opcao = MenuOpcao(opcao);                  //imprime na tela a funcao que o usuario selecionou         switch(opcao){                       case 1 : { //CASE 1 = Insere um numero na arvore                            Cabecalho();                            printf("Opcao [1] - Insere numero na arvore\n\n");                                                       printf("Digite -1 para sair\n\n");                            do{                               printf("Inserir o numero: ");                               scanf("%d", &numero);                                                              if(numero!=-1){                                         x.Chave = numero;                                         Insere(x, &Dicionario);                                                           }                             }while (numero!=-1);                                                       break;                       }                       case 2 : {//CASE 2 = Imprime na tela a ordem crescente da arvore                            Cabecalho();                            printf("Opcao [2] - Imprime em ordem crescente\n\n");                                                       OrdemCrescente(Dicionario);                            break;                       }                       case 3 : {//CASE 3 = Imprime na tela a ordem decrescente da arvore                            Cabecalho();                            printf("Opcao [3] - Imprime em ordem decrescente\n\n");                                                       OrdemDecrescente(Dicionario);                            break;                       }                       case 4 : {//CASE 4 = Imprime o maior numero na tela                            Cabecalho();                            printf("Opcao [4] - Imprime do maior numero da arvore\n\n");                                                       MaiorNumero(Dicionario);                                                       break;                       }                       case 5 : {//CASE 5 = Imprime o menor numero na tela                            Cabecalho();                            printf("Opcao [5] - Imprime do maior numero da arvore\n\n");                                                       MenorNumero(Dicionario);                                                       break;                       }                       case 6 : {//CASE 6 = Imprime na tela os nos folhas                            Cabecalho();                            printf("Opcao [6] - Imprime do nos folhas\n\n");                                                       NoFolhas(Dicionario);                                                       break;                       }                       case 7 : {//CASE 7 = Imprime na tela os numero a direita do numero informado pelo usuario                            Cabecalho();                            printf("Opcao [7] - Imprime sub-arvore direita de um no\n\n");                                                       printf("\n\nDigite um numero: ");                            scanf("%d",&numero);                                                       ProcuraNumeroDir(Dicionario, numero);                                                       break;                       }                       case 8 : {//CASE 8 = Imprime na tela os numero a esquerda do numero informado pelo usuario                            Cabecalho();                            printf("Opcao [8]\n\n");                                                                                  printf("\n\nDigite um numero: ");                            scanf("%d",&numero);                                                       ProcuraNumeroEsq(Dicionario, numero);                                                       break;                       }                       case 9 : {//CASE 9 = Imprime na tela a arvore binaria                            Cabecalho();                            printf("Opcao [9] - Imprime a arvore binaria\n\n");                                                       ImprimeArvore(0, Dicionario);                                                       break;                       }                       case 0 : {//CASE 9 = Imprime na tela a arvore binaria                            exit (0);                            break;                       }                       default :                               Cabecalho();                               printf(">>> ERRO <<<\n");                               printf("\n\nOpcao invalida.\n\n");                                        }        printf("\nPresione ENTER para voltar ao MENU\n\n");        system("PAUSE");     }       } // funcoes // Inicializa a arvore void Inicializa(Apontador *p){     *p = NULL; } double rand0a1(){   double resultado = (double) rand()/ RAND_MAX; /* Dividir pelo maior inteiro */   if(resultado>1.0)     resultado = 1.0;     return resultado; } // faz o embaralhamento dos numeros void Permut( TipoChave A[], int n) {   int i,j;   TipoChave b;   for(i = n; i>0; i --){       j = (i * rand0a1());       b = A[i];       A[i] = A[j];       A[j] = b;     } } // insere na arove os dados informados void Insere(Registro x, Apontador *p){     if (*p == NULL){         *p = (Apontador)malloc(sizeof(No));         (*p)->Reg = x;         (*p)->Esq = NULL;         (*p)->Dir = NULL;         return;     }     if (x.Chave < (*p)->Reg.Chave){         Insere(x, &(*p)->Esq);         return;     }     if (x.Chave > (*p)->Reg.Chave)         Insere(x, &(*p)->Dir);         else             printf("Erro : Registro já existe na arvore\n"); } void TestaI(No *p, int pai){     if (p == NULL)     return;     if (p->Esq != NULL){         if (p->Reg.Chave < p->Esq->Reg.Chave){             printf("Erro: Pai %ld menor que filho a esquerda %ld\n", p->Reg.Chave, p->Esq->Reg.Chave);             exit(1);         }     }     if (p->Dir != NULL){         if (p->Reg.Chave > p->Dir->Reg.Chave){             printf("Erro: Pai %ld maior que filho a direita %ld\n",  p->Reg.Chave, p->Dir->Reg.Chave);             exit(1);         }     }     TestaI(p->Esq, p->Reg.Chave);     TestaI(p->Dir, p->Reg.Chave); } void Testa(No *p){     if (p != NULL)         TestaI(p, p->Reg.Chave); } //-------------- Cabecalho ------------------------------------------- void Cabecalho(){      system("cls");           printf("#################################################\n");      printf("#\t\t   AEDIII\t\t\t#\n");      printf("#-----------------------------------------------#\n");      printf("#\t\tArvore Binaria\t\t\t#\n");      printf("#################################################\n\n"); } //-------------- Menu de opcao ------------------------------------------- int MenuOpcao(int opcao){            Cabecalho();     printf(">> Menu - Escolha umas das opcoes abaixo\n\n");          printf("1 -> Inserir numero\n");     printf("2 -> Imprime ordem crescente\n");     printf("3 -> Imprime ordem decrescente\n");     printf("4 -> Imprime maior numero\n");     printf("5 -> Imprime menor numero\n");     printf("6 -> Imprime nos folhas\n");     printf("7 -> Imprime sub-arvore direita de um no\n");     printf("8 -> Imprime sub-arvore esquerda de um no\n");     printf("9 -> Imprime arvore\n");     printf("0 -> Finalizar o programa\n\n");                  printf("Digite uma opcao: ");     scanf("%d", &opcao);          return opcao;   } //-------------- Imprime a ordem crescente da arvore ------------------- void OrdemCrescente(No *p){     if (p == NULL)         return;          OrdemCrescente(p->Esq);     printf("> %d\n", p->Reg.Chave);     OrdemCrescente(p->Dir);      } //-------------- Imprime a ordem decrescente da arvore ------------------- void OrdemDecrescente(No *p){     if (p == NULL)         return;          OrdemDecrescente(p->Dir);     printf("> %d\n", p->Reg.Chave);     OrdemDecrescente(p->Esq);      } //--------------Imprime a arvore na tela --------------------------------- void ImprimeArvore(int camada, No *p){      int i;           if(p == NULL)         return;               ImprimeArvore(camada+1, p->Dir);           for(i=0;i<camada;i++)         printf("      ");               printf("%d\n",p->Reg.Chave);      ImprimeArvore(camada+1, p->Esq); } //--------------Procura o maior numero da arvore-------------------------- void MaiorNumero(No *p){     if((p->Dir != NULL) && (p->Dir->Reg.Chave > p->Reg.Chave))         MaiorNumero(p->Dir);         else             printf("Maior numero encontrado foi: %d\n\n", p->Reg.Chave); } //--------------Procura o menor numero da arvore-------------------------- void MenorNumero(No* p){     if((p->Esq != NULL) && (p->Esq->Reg.Chave < p->Reg.Chave))          MenorNumero(p->Esq);          else              printf("Menor numero encontrado foi: %d\n\n", p->Reg.Chave); } //-----------------Procura os nos folhas da arvore------------------------ void NoFolhas(No *p){    if(p == NULL)         return;             NoFolhas(p->Dir);       if(p->Esq == NULL && p->Dir == NULL)         printf("%d\n",p->Reg.Chave);             NoFolhas(p->Esq);    } //--------------Procura os nos a Direita de numero selecionado------------ void ProcuraNumeroDir(No *p, int numero){    if (p == NULL)        return;           if (numero == p->Reg.Chave)        OrdemCrescente(p->Dir);    if (numero < p->Reg.Chave)        return ProcuraNumeroDir(p->Esq, numero);        else            return ProcuraNumeroDir(p->Dir, numero);                            } //--------------Procura os nos a Esquerda de numero selecionado------------ void ProcuraNumeroEsq(No *p, int numero){    if (p == NULL)        return;           if (numero == p->Reg.Chave)        OrdemDecrescente(p->Esq);    if (numero < p->Reg.Chave)        return ProcuraNumeroEsq(p->Esq, numero);        else            return ProcuraNumeroEsq(p->Dir, numero);                            }[/code]

Espero que ajude!!

Também fiz um sobre tabela HASH semana passada se quiser eu posto ele aqui!!!

Editado por jrsilvamg
Link para o comentário
Compartilhar em outros sites

Participe da discussão

Você pode postar agora e se registrar depois. Se você já tem uma conta, acesse agora para postar com sua conta.

Visitante
Responder esta pergunta...

×   Você colou conteúdo com formatação.   Remover formatação

  Apenas 75 emoticons são permitidos.

×   Seu link foi incorporado automaticamente.   Exibir como um link em vez disso

×   Seu conteúdo anterior foi restaurado.   Limpar Editor

×   Você não pode colar imagens diretamente. Carregar ou inserir imagens do URL.



  • Estatísticas dos Fóruns

    • Tópicos
      152,3k
    • Posts
      652,4k
×
×
  • Criar Novo...