Documente Academic
Documente Profesional
Documente Cultură
Arbori Cu Radacina
Arbori Cu Radacina
Se numete nlimea unui arbore diferena dintre nivelul maxim din arbore i nivelul minim (nivelul rdcinii).
Pentru arborele de mai sus, nlimea este 4-1=3.
Observaie: ntr-un arbore orice nod este legat de rdcin printr-un lan unic de lungime minim (corespunztor
parcurgerii BF de la grafuri).
2
Nodurile unui arbore oarecare pot fi reprezentate sub forma unei zone de memorie alocat dinamic sau sub
forma unui tablou de date alocate static. Fie urmtorul arbore:
4
int i;
if(p)
{
for(i=0;i<p->n;i++)
if(p->leg[i]) Postordine(p->leg[i]);
cout<<p->inf<<" "; //afisez nodul tata
//afisez descendentii
}
}
void Adauga(ARB *p) //adauga un nod in coada
{
if(prim>ultim) cout<<"Coada este plina\n";
else coada[ultim++]=p;
}
ARB* Extrage_nod() //extrage un nod din coada
{
if(prim==ultim) return 0;
else return coada[prim++];
}
void Traversare_nivele(ARB *rad)
{
ARB *p; int i;
prim=ultim=0;
Adauga(rad); //in coada se introduce nodul radacina
do{
p=Extrage_nod(); //extrag un nod din coada
if(p)
{
cout<<p->inf<<" "; //afisez informatia nodului
for(i=0;i<p->n;i++)
Adauga(p->leg[i]); //adaug in coada descendentii nodului
}
}while(p);
cout<<"\n";
}
void Sterge(ARB *p) //stergerea unui nod din arbore
{
int i;
if(p)
for(i=0;i<p->n;i++) Sterge(p->leg[i]);
//mai intai sterg descendentii nodului
delete p; //sterg nodul tata
}
int main()
{
ARB *rad; //radacina arborelui oarecare
cout<<"\n\t\tIntroduceti arborele:\n";
rad=Creare(); //arborele a fost creat
cout<<"\n\t\tTraversarea in preordine:\n";
Preordine(rad);
cout<<"\n\t\tTraversarea in postordine:\n";
Postordine(rad);
cout<<"\n\t\tTraversarea pe nivele:\n";
Traversare_nivele(rad);
Sterge(rad); //stergerea arborelui din heap
return 0;
}
5
Definiie: Un arbore binar este o mulime finit de noduri care este fie vid, fie reprezint un arbore ordonat n
care fiecare nod are cel mult doi descendeni.
Un arbore binar conine cel mult doi subarbori, pe care i numim subarbore stng, respectiv subarbore
drept. Ei se pot obine prin suprimarea rdcinii i a nodurilor incidente cu aceasta. Un nod fr descendeni se
numete nod terminal sau frunz.
Un arbore binar n care fiecare nod are 0 sau 2 descendeni se numete arbore binar complet. Un astfel de
arbore apare n figura de mai jos.
S[i]
D[i]
2
3
4
5
0
0
0
0
6
7
0
0
8
9
0
0
0
0
Se observ c nu este important s se precizeze rdcina, deoarece ea nu este descendenta nici unui nod.
b) Legturi de tip tat. Se folosesc doi vectori: TATA i DESC. Pentru fiecare nod i, TATA[i] precizeaz
care nod i este ascendent (nodul printe). DESC[i] poate lua dou valori: -1 dac i este descendent stng pentru
TATA[i] i 1 dac este descendent drept pentru acesta. Pentru nodul rdcin, care nu are un nod printe asociat,
valoarea corespunztoare n cei doi vectori este 0. Pentru arborele binar de mai sus, configuraia celor doi vectori
este:
Nodul i
TATA[i]
DESC[i]
1
0
0
2
1
-1
3
1
1
4
2
-1
5
2
1
6
5
-1
7
5
1
8
7
-1
9
7
1
7
{
if(S[k]) SRD(S[k]);
cout<<k<<" ";
if(D[k]) SRD(D[k]);
}
void RSD(int k) //parcurgere preordine
{
cout<<k<<" ";
if(S[k]) RSD(S[k]);
if(D[k]) RSD(D[k]);
}
void SDR(int k) //parcurgere postordine
{
if(S[k]) SDR(S[k]);
if(D[k]) SDR(D[k]);
cout<<k<<" ";
}
int main()
{
int i;
cout<<"numar varfuri, n=";
do{
cin>>n;
}while(n<1||n>N);
cout<<"radacina arborelui: ";
do{
cin>>rad;
}while(rad<1||rad>n);
//memorarea arborelui prin vectorii S,D
for(i=1;i<=n;i++)
{
cout<<"descendentii stanga,dreapta ("<<i<<"): ";
cin>>S[i]>>D[i];
}
cout<<"\n\tParcurgerea in preordine:\n";
RSD(rad);
cout<<"\n\n\tParcurgerea in inordine:\n";
SRD(rad);
cout<<"\n\n\tParcurgerea in postordine:\n";
SDR(rad);
cout<<"\n";
return 0;
}
8
-
un descendent vid trebuie marcat printr-o proprietate stabilit asupra informaiei (spre exemplu apariia
valorii 0 drept coninut informaional al nodului curent)
Spre exemplu, pentru arborele binar urmtor
irul datelor furnizat la intrare va arta astfel: 1 2 4 0 0
5 7 0 0 0 3 0 6 8 0 0 9 0 0
ceea ce
corespunde unei liste n preordine a nodurilor.
Funcia care creaz un arbore binar furnizat ca mai sus
este:
9
void SRD(ARB *r) //parcurgere inordine
{
if(r)
{
SRD(r->st); cout<<r->inf<<" "; SRD(r->dr);
}
}
void RSD(ARB *r) //parcurgere preordine
{
if(r)
{
cout<<r->inf<<" "; RSD(r->st); RSD(r->dr);
}
}
void SDR(ARB *r) //parcurgere postordine
{
if(r)
{
SDR(r->st); SDR(r->dr); cout<<r->inf<<" ";
}
}
void Sterge(ARB *r)
{
if(r)
{
Sterge(r->st); Sterge(r->dr); delete r;
}
}
int main()
{
ARB *rad;
//adresa radacinii arborelui
cout<<"Introduceti nodurile : ";
Creare(rad); //crearea arborelui binar
cout<<"\nNodurile in inordine:
"; SRD(rad);
cout<<"\nNodurile in preordine: "; RSD(rad);
cout<<"\nNodurile in postordine: "; SDR(rad);
cout<<"\n";
Sterge(rad); //stergerea arborelui din heap
return 0;
}
10
Exemplu: Fie expresia E=a*b+c/d-e. Ne propunem s determinm arborele binar asociat. Se observ c,
deoarece nu este stabilit modul de asociere a termenilor, se pot obine doi arbori binari, dup cum se consider
E1=a*b+c/d i E2=e, sau E1=a*b i E2=c/d-e. Se obin corespunztor arborii din figura de mai jos.
11
void Creare(ARB* &c,int li,int ls,char epf[],int pfp[])
{
int i,j,min;
min=pfp[ls]; i=ls;
for(j=ls;j>=li;j--)
if(pfp[j]<min) { min=pfp[j]; i=j; }
c=new ARB; c->inf=efp[i];
if(li==ls)
c->st=c->dr=0;
else
{
Creare(c->st,li,i-1,efp,pfp);
Creare(c->dr,i+1,ls,efp,pfp);
}
}
void Parcurgere(ARB *c)
{
if(c)
{
cout<<c->inf;Parcurgere(c->st); Parcurgere(c->dr);
}
}
int main()
{
int i,j=0;
cout<<"introduceti expresia: "; gets(e);
for(i=0;e[i];i++)
switch(e[i])
{
case ')': j-=10; break;
case '(': j+=10; break;
case '+':
case '-': p[i]=j+1; break;
case '*':
case '/': p[i]=j+10; break;
default: p[i]=1000;
}
j=-1;
for(i=0;e[i];i++)
if(e[i]!=')'&&e[i]!='(')
{
j++; efp[j]=e[i]; pfp[j]=p[i];
}
Creare(rad,0,j,efp,pfp);
cout<<"\nForma poloneza prefixata este: ";
Parcurgere(rad); cout<<"\n";
return 0;
}
Modul n care se folosete forma polonez prefixat pentru evaluarea expresiei aritmetice asociate este
urmtorul: se parcurge forma polonez prefixat de la dreapta la stnga utilizndu-se o stiv pentru memorarea
operanzilor. Dac simbolul curent este un operand, el se introduce n vrful stivei. Dac simbolul citit este un
operator, se aplic acest operator primilor doi operanzi din vrful stivei obinndu-se un rezultat. Cei doi operanzi
sunt eliminai din stiv i n locul lor se trece rezultatul obinut. Se trece apoi la urmtorul simbol care conine forma
polonez prefixat asociat expresiei aritmetice.
Programul urmtor ilustreaz aceast modalitate de evaluare a expresiilor aritmetice. Operanzii sunt
identificai printr-o singur liter, iar operatorii sunt binari din mulimea +,-,*,/.
#include<iostream>
#include<string.h>
using namespace std;
12
char e[100];
float s[100],x,y; //s este stiva
int n,i,p;
int main()
{
cout<<"\tIntroduceti expresia aritmetica in forma prefixata\n";
gets(e); n=strlen(e);
for(i=n-1;i>=0;i--)
if(e[i]=='+'||e[i]=='-'||e[i]=='*'||e[i]=='/')
{
x=s[p--]; y=s[p];
switch(e[i])
{
case '+': s[p]=x+y; break;
case '-': s[p]=x-y; break;
case '*': s[p]=x*y; break;
case '/': s[p]=x/y; break;
}
}
else
s[++p]=e[i]-'0';
cout<<"\nvaloarea expresiei este "<<s[1]<<endl;
return 0;
}