[ Back ] [ Up ]
- The specifications.
Let T be a structured tree and S be its node.
Below we present a list of variables, and instrumental procedures that
will be used for description of the algorithms.
- Denotations and definitions.
With respect to the current node S a sweep
<Lb-1,lb-1,…,L0,l0,S,r0,R0,…,rb-1,Rb-1>
composed of b neighbors and their delimiting keys to the left
and to the right of S is called the vicinity of S, where
 | S is the current node; |
 | Li denotes the i-th left neighbor of S; |
 | Ri denotes the i-th right neighbor of S; |
 | li is the delimiting key of nodes Li and Li-1; |
 | ri is the delimiting key of nodes Ri and Ri-1. |
 | F denotes the direct parent of S. |
 | FLi (FRi) denotes the parent of node Li
(Ri). |
 | Fli (Fri) denotes the node containing the delimiting
key li (ri). |
 | Li, li, Ri, ri, FLi, FRi,
Fli, Fri are the local variables of the procedures. |
 | WW(S) means that M(S) <= p |
 | WC(S) means that |S| >= q |
 | WF(S) means that WW(S) and WC(S) |
 | IC(s) means that sweep s is incompressible |
 | WB(s) means that IC(s) for any sweep sof length
b containing S |
 | Sweepbm(S) denotes the m-th (m = 0,…,b)
sweep of length b containing node S, namely
Sweepb0(S) = <Lb-1,lb-1,…,L0,l0,S>
Sweepbb(S) = <S,r0,R0,…,rb-1,Rb-1>
Sweepbm(S) = <Lb-m-1,lb-m-1,…,L0,l0,S,r0,R0,…,rm-1,Rm-1>
|
- Instrumental procedures.
The following procedures and functions are used for describing the
algorithms.
 | Search(T,k) given a S(b)-tree T and key k verifies whether the key
is contained in the tree. The result of the function is (BoolValue, S, ki).
BoolValue is a Boolean variable that specifies whether k is found in the
tree or not. S is the node that was visited last while searching in the tree, and ki
is the minimal key of S that is greater than or equal to the given key k. |
 | MakeVicinity(S) initializes the local variables Li, li,
Ri, ri, FLi, FRi, Fli, Fri
according to S. |
 | Replace(S,P,Q) replaces a part of node S, that coincides with P,
with Q. |
 | Functions LeftOf(S,k) and RightOf(S,k) define two parts of S, that
are, respectively, to the left and to the right of key k in S. |
 | For each m = 0,…,b we define a function Compressm that is
applied to not incompressible sweeps of length b. The result of the function is an
incompressible sweep of length b-1, composed of b well-formed (WF see
above) nodes. Namely, let
Compressm(A0,a1,…,ab,Ab)
= <C0,c1,…,cb-1,Cb-1>
then the resulting sweep is obtained by distributing the contents of node Am
between the other nodes of the initial sweep in such a way that ai
< ci for i = 1,…,m, and ci < ai+1
for i = m,…,b-1. |
 | ComputeSets(S) calculates seven subsets of the set of keys of node S: MLeft,
MRight, MDelimL, MDelimR, MDelim, coMLeft, and coMRight. |
 | ChooseAny(E), ChooseMax(E), ChooseMin(E) for the given key set E return,
respectively, an arbitrary, the maximal, and the minimal key of set E. |
 | MinKey(S) looks for the minimal key in the sub-tree rooted at S, returns (S',
k') where k' is the minimal key, and S' is the leaf that contains k'. |
 | GlueParents(F) takes the parent F of the current node S, the left,
and the right neighbors of F, glues them into one node, and returns the result. A
more sophisticated variant of this function is to check before gluing whether the
neighbors have been modified, and glue to F only the modified ones. |
 | CreateNewRoot(S) creates the new root of the tree, containing the only reference
to the given node S. |
 | ReleaseRoot(). If the root of the tree contains only one reference to a node S,
and no keys, then this root is removed, and S is assigned to be the new root of the
tree. |
 | [P,d,Q] creates a new node composed of all keys, and node references (in case of
internal nodes), of the given nodes P and Q, with the key d between
them. |
If k(S) denotes the set of all keys of node S, then the
definitions of the subsets are as follows.
MLeft = {d from k(S) such that IC(Lb-1,lb-1,…,L0,l0,LeftOf(S,d))}
MRight = {d from k(S) such that IC(RightOf(S,d),r0,R0,…,rb-1,Rb-1)}
MDelimL = {d from k(S) such that WF(RightOf(S,d))}
MDelimR = {d from k(S) such that WF(LeftOf(S,d))}
MDelim = MDelimL # MDelimR
coMLeft = k(S) \ MLeft
coMRight = k(S) \ MRight
The sets, and the ComputeSets() procedure are used in Balance_W()
to control the process of computation.
- The main procedures.
Procedure Insert(T,k)
(BoolValue, S, ki) = Search(T,k);
if BoolValue = True then return fi;
/* k haven't been found means that S is a leaf */
Replace(S,<ki>,<k,ki>) /* insert k
before ki */
Balance(S); /* balance T starting from leaf S */
EndProcedure
Procedure Delete(T,k)
(BoolValue, S, ki) = Search(T,k);
if BoolValue = False then return fi;
if S is not a leaf then
/* S is internal, and contains reference */
/* Si+1, which is to the right of ki in
S */
(S', k') = MinKey(Si+1);
Replace(S,<ki>,<k'>);
S := S';
k := k';
fi
/* S is a leaf */
Replace(S,<k>,<>) /* delete k from leaf S */
Balance(S); /* balance T starting from leaf S */
EndProcedure
Procedure Balance(S)
while S is not the root do
if ! WB(S) then F := Balance_B(S);
else if ! WC(S) then F := Balance_C(S);
else if ! WW(S) then F := Balance_W(S);
else S := F; continue;
fi fi fi
S := GlueParents(F);
od
/* S is the root now */
if |S| = 0 then
ReleaseRoot();
fi
if ! WW(S) then
CreateNewRoot(S);
Balance_W(S);
fi
EndProcedure
Procedure Balance_W(S)
MakeVicinity(S);
ComputeSets(S);
// Compression:
// Distribute all keys of S between
// the other nodes of the vicinity
if coMLeft # coMRight != Empty then
d := ChooseAny(coMLeft # coMRight);
<Xb-1,xb-1,…,x1,X0> :=
Compressb(Lb-1,lb-1,…,L0,l0,LeftOf(S,d));
<Y0,y1,…,yb-1,Yb-1> :=
Compress0(RightOf(S,d),r0,R0,…,rb-1,Rb-1);
for i := 1 to b-1 do
Replace(FLi,<Li>,<Xi>);
Replace(Fli,<li>,<xi>);
Replace(FRi,<Ri>,<Yi>);
Replace(Fri,<ri>,<yi>);
od
Replace(FR0,<R0>,<Y0>);
Replace(Fr0,<r0>,<d>);
if FL0 = F then
Replace(F,<L0,l0,S>,<X0>);
else
Replace(F,<S>,<X0>);
Replace(Fl0,<l0>,<x1>);
Replace(FL0,<X1,x1,L0>,<X1>);
fi
return F;
fi
// Move left:
// Move to the left part of the vicinity
// as much keys from S as possible
if coMLeft != Empty then
if MDelimL # coMLeft != Empty then d := ChooseAny(MDelimL #
coMLeft);
else d :=
ChooseMax(coMLeft);
<Xb-1,xb-1,…,x1,X0> :=
Compressb(Lb-1,lb-1,…,L0,l0,LeftOf(S,d));
Y := RightOf(S,d);
for i := 1 to b-1 do
Replace(FLi,<Li>,<Xi>);
Replace(Fli,<li>,<xi>);
od
Replace(FL0,<L0>,<X0>);
Replace(Fl0,<l0>,<d>);
Replace(F,<S>,<Y>);
if WW(Y) then return F; fi
S := Y;
MakeVicinity(S);
ComputeSets(S);
fi
// Move right:
// Move to the right part of the vicinity
// as much keys from S as possible
if coMRight != Empty then
if MDelimR # coMRight != Empty then d := ChooseAny(MDelimR
# coMRight);
else d :=
ChooseMin(coMRight);
X := LeftOf(S,d);
<Y0,y1,…,yb-1,Yb-1> :=
Compress0(RightOf(S,d),r0,R0,…,rb-1,Rb-1);
for i := 1 to b-1 do
Replace(FRi,<Ri>,<Yi>);
Replace(Fri,<ri>,<yi>);
od
Replace(FR0,<R0>,<Y0>);
Replace(Fr0,<r0>,<d>);
Replace(F,<S>,<X>);
if WW(X) then return F; fi
S := X;
MakeVicinity(S);
ComputeSets(S);
fi
// Split:
// Even after moving out of S all keys that fit into the other nodes
// of the vicinity S is still too large, and
// need to be split into two nodes
if MDelim != Empty then d := ChooseAny(MDelim);
else d := ChooseMax(MDelimR);
X := LeftOf(S,d);
Y := RightOf(S,d);
Replace(F,<S>,<X,d,Y>);
if WW(Y) then return F; fi
// Recursion:
// Even after splitting off a node X the remaining part Y is still
// too large, Balance_W() will be applied to Y again
F := Balance_W(Y);
return F;
EndProcedure
Procedure Balance_B(S)
MakeVicinity(S);
// First Compression:
// Check which of the first b sweeps of the vicinity is
// compressible, and compress the first found
for m := 0 to b-1 do
if ! IC(Sweepbm(S)) then break; fi
od
if m < b then // a compressible sweep is found, compress it
<Xb-m-1,xb-m-1,…,x1,X0,y0,Y0,y1,…,ym-1,Ym-1>
:= Compressb-m(Sweepbm(S));
for i := 1 to b-m-1 do
Replace(FLi,<Li>,<Xi>);
Replace(Fli,<li>,<xi>);
od
for i := 0 to m-1 do
Replace(FRi,<Ri>,<Yi>);
Replace(Fri,<ri>,<yi>);
od
if FL0 = F then
Replace(F,<L0,l0,S>,<X0>);
else
Replace(F,<S>,<X0>);
Replace(Fl0,<l0>,<x1>);
Replace(FL0,<X1,x1,L0>,<X1>);
fi
m := m + 1;
S := X1;
MakeVicinity(S);
fi
// Second Compression:
// Among the remaining sweeps of the vicinity check which one is
// compressible, and compress the first found
for m := m to b do
if ! IC(Sweepbm(S)) then break; fi
od
if m > b then return F; fi
// A compressible sweep is found, compress it
<Xb-m-1,xb-m-1,…,x1,X0,y0,Y0,y1,…,ym-1,Ym-1>
:= Compressb-m(Sweepbm(S));
for i := 0 to b-m-1 do
Replace(FLi,<Li>,<Xi>);
Replace(Fli,<li>,<xi>);
od
for i := 1 to m-1 do
Replace(FRi,<Ri>,<Yi>);
Replace(Fri,<ri>,<yi>);
od
if FR0 = F then
Replace(F,<S,r0,R0>,<Y0>);
else
Replace(F,<S>,<Y0>);
Replace(Fr0,<r0>,<y1>);
Replace(FR0,<R0,y1,Y1>,<Y1>);
fi
// It is proven that not more than 2 nodes of the vicinity
// can be shrunken
return F;
EndProcedure
Procedure Balance_C(S)
MakeVicinity(S);
// If S has too few keys we glue it with one of the nearest
// neighbors, and apply Balance_W to the result if necessary
if |F| > 0 then // at least one key is in F
if FL0 = F then
X := [L0,l0,S];
Replace(F,<L0,l0,S>,<X>);
else // FR0 = F, since F contains at least one key
X := [S,r0,R0];
Replace(F,<S,r0,R0>,<X>);
fi
else // there is no keys in F only the reference to S
if |FL0| > 1 then
X := [L0,l0,S];
Replace(F,<S>,<X>);
Replace(Fl0,<l0>,<l1>);
Replace(FL0,<L1,l1,L0>,<L1>);
else /* |FR0| > 1 */
X := [L0,l0,S];
Replace(F,<S>,<X>);
Replace(Fr0,<r0>,<r1>);
Replace(FR0,<R0,r1,R1>,<R1>);
fi
fi
if ! WW(X) then
F := Balance_W(Y);
fi
return F;
EndProcedure
|