Правильно ли расщепляются узлы в моем 2-3 дереве?
Я ввел 3 значения в дерево, которое я создал, но я могу найти только значение корневого узла не выше или ниже. Что-то не так с моим кодом или я должен начать все сначала, потому что я все понял неправильно?
Заранее спасибо всем, у кого есть время помочь с этим.
Я проверил свое дерево с:
TwoThreeTree MyTree;
MyTree.Add(20);
MyTree.Add(10);
MyTree.Add(60);
if ( MyTree.Search(20) ) return 0;
return 1;
И код для моего дерева:
Заголовок:
enum LOCATION {
LOW,
HIGH,
MIDDLE
};
class TwoThreeTree {
private:
class TwoThreeNode {
private:
// Currently not needed
public:
TwoThreeNode();
~TwoThreeNode();
int Data1, Data2;
bool TwoData; // Data2 is used or not?
TwoThreeNode *Child1, *Child2, *Child3;
TwoThreeNode* Parent;
};
TwoThreeNode* Root;
public:
TwoThreeTree();
~TwoThreeTree();
void Add(int item);
bool Search(int item);
void Split(TwoThreeNode* Leaf, int item, LOCATION location, TwoThreeNode* Lower, TwoThreeNode* Higher);
void KillNodes(TwoThreeNode* Node);
int Highest(int a, int b, int c);
int Lowest(int a, int b, int c);
int Middle(int a, int b, int c);
};
Источник:
#include "TwoThreeTree.h"
TwoThreeTree::TwoThreeTree() :
Root(0)
{
}
TwoThreeTree::~TwoThreeTree()
{
KillNodes( Root );
}
void TwoThreeTree::Split(TwoThreeNode* Leaf, int item, LOCATION location, TwoThreeNode* Lower, TwoThreeNode* Higher)
{
if ( !Leaf && Lower && Higher )
{
TwoThreeNode* NewRoot = new TwoThreeNode;
NewRoot->Data1 = item;
NewRoot->Child1 = Lower;
NewRoot->Child2 = Higher;
Lower->Parent = NewRoot;
Higher->Parent = NewRoot;
Root = NewRoot;
}
else if ( Leaf && !Lower && !Higher )
{
TwoThreeNode* hChild = new TwoThreeNode; // higher child
int Med = Middle( Leaf->Data1, Leaf->Data2, item );
int High = Highest( Leaf->Data1, Leaf->Data2, item );
int Low = Lowest( Leaf->Data1, Leaf->Data2, item );
Leaf->Data2 = 0; Leaf->TwoData = false;
Leaf->Data1 = Low;
hChild->Data1 = High;
Split ( Leaf->Parent, Med, location, Leaf, hChild );
}
else if ( Leaf && Lower && Higher )
{
if ( Leaf->TwoData )
{
if ( location == HIGH )
{
TwoThreeNode* hChild = new TwoThreeNode; // higher child
int pHigh = Highest( Leaf->Data1, Leaf->Data2, item );
int pLow = Lowest( Leaf->Data1, Leaf->Data2, item );
int pMed = Middle( Leaf->Data1, Leaf->Data2, item );
hChild->Data1 = pHigh;
hChild->Child1 = Lower;
hChild->Child2 = Higher;
Lower->Parent = hChild;
Higher->Parent = hChild;
Leaf->Data2 = 0; Leaf->TwoData = false;
Leaf->Data1 = pLow;
Leaf->Child3 = 0;
Split( Leaf->Parent, pMed, location, Leaf, hChild );
}
else if ( location == LOW )
{
TwoThreeNode* lChild = new TwoThreeNode; // lower child
int pHigh = Highest( Leaf->Data1, Leaf->Data2, item );
int pLow = Lowest( Leaf->Data1, Leaf->Data2, item );
int pMed = Middle( Leaf->Data1, Leaf->Data2, item );
lChild->Data1 = pLow;
lChild->Child1 = Lower;
lChild->Child2 = Higher;
Lower->Parent = lChild;
Higher->Parent = lChild;
Leaf->Data2 = 0; Leaf->TwoData = false;
Leaf->Data1 = pHigh;
Leaf->Child1 = Leaf->Child2;
Leaf->Child2 = Leaf->Child3;
Leaf->Child3 = 0;
Split( Leaf->Parent, pMed, location, lChild, Leaf );
}
else if ( location == MIDDLE )
{
TwoThreeNode* hChild = new TwoThreeNode;
int pHigh = Highest( Leaf->Data1, Leaf->Data2, item );
int pLow = Lowest( Leaf->Data1, Leaf->Data2, item );
int pMed = Middle( Leaf->Data1, Leaf->Data2, item );
hChild->Data1 = pHigh;
Leaf->Data1 = pLow;
Leaf->TwoData = false;
hChild->Child2 = Leaf->Child3;
Leaf->Child3 = 0;
hChild->Child1 = Higher;
Leaf->Child2 = Lower;
Higher->Parent = hChild;
Lower->Parent = Leaf;
Split( Leaf->Parent, pMed, location, Leaf, hChild );
}
}
else // if parent isn't full
{
if ( location == HIGH )
{
Leaf->Child2 = Lower;
Leaf->Child3 = Higher;
}
else if ( location == LOW )
{
Leaf->Child3 = Leaf->Child2;
Leaf->Child1 = Lower;
Leaf->Child2 = Higher;
}
Lower->Parent = Leaf;
Higher->Parent = Leaf;
if ( Leaf->Data1 > item )
{
Leaf->Data2 = Leaf->Data1;
Leaf->Data1 = item;
}
else
Leaf->Data2 = item;
}
}
}
void TwoThreeTree::Add(int item)
{
if ( !Root ) {
Root = new TwoThreeNode;
Root->Data1 = item;
}
else if ( !Root->TwoData && ( !Root->Child1 && !Root->Child2 && !Root->Child3))
{
if ( Root->Data1 > item )
{
Root->Data2 = Root->Data1;
Root->Data1 = item;
}
else
Root->Data2 = item;
}
else
{
LOCATION loc;
TwoThreeNode* tNode;
TwoThreeNode* Leaf = tNode = Root;
while ( tNode )
{
Leaf = tNode;
if ( tNode->TwoData )
{
if ( Highest( tNode->Data1, tNode->Data2, item ) == item ) {
tNode = tNode->Child3;
loc = HIGH;
}
else if ( Lowest( tNode->Data1, tNode->Data2, item ) == item ) {
tNode = tNode->Child1;
loc = LOW;
}
else {
tNode = tNode->Child2;
loc = MIDDLE;
}
}
else
{
if ( item > tNode->Data1 ) {
tNode = tNode->Child2;
loc = HIGH;
}
else {
tNode = tNode->Child1;
loc = LOW;
}
}
}
if ( Leaf->TwoData ) {
Split( Leaf, item, loc, 0, 0 );
}
else {
if ( Leaf->Data1 > item ) {
Leaf->Data2 = Leaf->Data1;
Leaf->Data1 = item;
}
else
Leaf->Data2 = item;
}
}
}
bool TwoThreeTree::Search(int item)
{
TwoThreeNode* tNode = Root;
while ( tNode )
{
if ( tNode->TwoData )
{
if ( item == tNode->Data1 || item == tNode->Data2 ) return true;
if ( Highest( tNode->Data1, tNode->Data2, item ) == item ) tNode = tNode->Child3;
else if ( Lowest( tNode->Data1, tNode->Data2, item ) == item ) tNode = tNode->Child1;
else tNode = tNode->Child2;
}
else
{
if ( item == tNode->Data1 ) return true;
if ( item > tNode->Data1 ) tNode = tNode->Child2;
else tNode = tNode->Child1;
}
}
return false;
}
int TwoThreeTree::Highest( int a, int b, int c )
{
int highest = ( a > b ) ? a : b;
return ( highest > c ) ? highest : c;
}
int TwoThreeTree::Lowest( int a, int b, int c )
{
int lowest = ( a < b ) ? a : b;
return ( lowest < c ) ? lowest : c;
}
int TwoThreeTree::Middle( int a, int b, int c )
{
int high = Highest(a, b, c);
int low = Lowest(a, b, c);
if ( a != high && a != low ) return a;
return ( b != high && b != low ) ? b : c;
}
void TwoThreeTree::KillNodes( TwoThreeNode* Node )
{
if ( Node )
{
KillNodes( Node->Child1 );
KillNodes( Node->Child2 );
KillNodes( Node->Child3 );
delete Node;
}
}
TwoThreeTree::TwoThreeNode::TwoThreeNode() :
Child1(0), Child2(0), Child3(0),
Parent(0), TwoData(false)
{
}
TwoThreeTree::TwoThreeNode::~TwoThreeNode()
{
}