#ifdef AAPL_NAMESPACE
namespace
Aapl {
#endif
template
< BST_TEMPL_DECLARE >
class
BstTable :
public
Compare,
public
Vector< Element, Resize >
{
typedef
Vector<Element, Resize> BaseVector;
typedef
Table<Element> BaseTable;
public
:
BstTable() { }
BstTable(
const
Key &key)
{ insert(key); }
#if defined( BSTMAP )
BstTable(
const
Key &key,
const
Value &val)
{ insert(key, val); }
#endif
#if ! defined( BSTSET )
BstTable(
const
Element &el)
{ insert(el); }
#endif
Element *insert(
const
Key &key, Element **lastFound = 0);
Element *insertMulti(
const
Key &key);
bool
insert(
const
BstTable &other);
void
insertMulti(
const
BstTable &other);
#if defined( BSTMAP )
Element *insert(
const
Key &key,
const
Value &val,
Element **lastFound = 0);
Element *insertMulti(
const
Key &key,
const
Value &val );
#endif
#if ! defined( BSTSET )
Element *insert(
const
Element &el, Element **lastFound = 0);
Element *insertMulti(
const
Element &el);
#endif
Element *find(
const
Key &key, Element **lastFound = 0)
const
;
bool
findMulti(
const
Key &key, Element *&lower,
Element *&upper )
const
;
bool
remove
(
const
Key &key);
bool
remove
(Element *item);
long
removeMulti(
const
Key &key);
long
removeMulti(Element *lower, Element *upper);
#if !defined( SHARED_BST )
void
vinsert(
long
pos,
const
Element &val)
{ Vector< Element, Resize >::insert( pos, &val, 1 ); }
void
vinsert(
long
pos,
const
Element *val,
long
len)
{ Vector< Element, Resize >::insert( pos, val, len ); }
void
vinsert(
long
pos,
const
BstTable &v)
{ Vector< Element, Resize >::insert( pos, v.data, v.tabLen ); }
void
vremove(
long
pos)
{ Vector< Element, Resize >::
remove
( pos, 1 ); }
void
vremove(
long
pos,
long
len)
{ Vector< Element, Resize >::
remove
( pos, len ); }
#else /* SHARED_BST */
void
vinsert(
long
pos,
const
Element &val)
{ Vector< Element, Resize >::insert( pos, &val, 1 ); }
void
vinsert(
long
pos,
const
Element *val,
long
len)
{ Vector< Element, Resize >::insert( pos, val, len ); }
void
vinsert(
long
pos,
const
BstTable &v)
{ Vector< Element, Resize >::insert( pos, v.data, v.length() ); }
void
vremove(
long
pos)
{ Vector< Element, Resize >::
remove
( pos, 1 ); }
void
vremove(
long
pos,
long
len)
{ Vector< Element, Resize >::
remove
( pos, len ); }
#endif /* SHARED_BST */
};
#if 0
#if defined( SHARED_BST )
template
<BST_TEMPL_DEF> BstTable<BST_TEMPL_USE>::
BstTable(
long
allocLen )
{
if
( allocLen > 0 ) {
STabHead *head = (STabHead*)
malloc
(
sizeof
(STabHead) +
sizeof
(Element) * allocLen );
if
( head == 0 )
throw
std::bad_alloc();
head->refCount = 1;
head->allocLen = allocLen;
head->tabLen = 0;
BaseTable::data = (Element*) (head + 1);
}
}
#else
template
<BST_TEMPL_DEF> BstTable<BST_TEMPL_USE>::
BstTable(
long
allocLen )
{
BaseTable::allocLen = allocLen;
if
( BaseTable::allocLen > 0 ) {
BaseTable::data = (Element*)
malloc
(
sizeof
(Element) * BaseTable::allocLen);
if
( BaseTable::data == NULL )
throw
std::bad_alloc();
}
}
#endif
#endif
template
<BST_TEMPL_DEF>
bool
BstTable<BST_TEMPL_USE>::
remove
(
const
Key &key)
{
Element *el = find(key);
if
( el != 0 ) {
Vector< Element >::
remove
(el - BaseTable::data);
return
true
;
}
return
false
;
}
template
<BST_TEMPL_DEF>
bool
BstTable<BST_TEMPL_USE>::
remove
( Element *item )
{
if
( item != 0 ) {
Vector< Element >::
remove
(item - BaseTable::data);
return
true
;
}
return
false
;
}
template
<BST_TEMPL_DEF>
long
BstTable<BST_TEMPL_USE>::
removeMulti(
const
Key &key)
{
Element *low, *high;
if
( findMulti(key, low, high) ) {
long
num = high - low + 1;
Vector< Element >::
remove
(low - BaseTable::data, num);
return
num;
}
return
0;
}
template
<BST_TEMPL_DEF>
long
BstTable<BST_TEMPL_USE>::
removeMulti(Element *lower, Element *upper)
{
long
num = upper - lower + 1;
Vector< Element >::
remove
(lower - BaseTable::data, num);
return
num;
}
template
<BST_TEMPL_DEF>
bool
BstTable<BST_TEMPL_USE>::
findMulti(
const
Key &key, Element *&low, Element *&high )
const
{
const
Element *lower, *mid, *upper;
long
keyRelation;
const
long
tblLen = BaseTable::length();
if
( BaseTable::data == 0 )
return
false
;
lower = BaseTable::data;
upper = BaseTable::data + tblLen - 1;
while
(
true
) {
if
( upper < lower ) {
return
false
;
}
mid = lower + ((upper-lower)>>1);
keyRelation =
this
->compare(key, GET_KEY(*mid));
if
( keyRelation < 0 )
upper = mid - 1;
else
if
( keyRelation > 0 )
lower = mid + 1;
else
{
Element *lowEnd = BaseTable::data - 1;
Element *highEnd = BaseTable::data + tblLen;
lower = mid - 1;
while
( lower != lowEnd &&
this
->compare(key, GET_KEY(*lower)) == 0 )
lower--;
upper = mid + 1;
while
( upper != highEnd &&
this
->compare(key, GET_KEY(*upper)) == 0 )
upper++;
low = (Element*)lower + 1;
high = (Element*)upper - 1;
return
true
;
}
}
}
template
<BST_TEMPL_DEF> Element *BstTable<BST_TEMPL_USE>::
find(
const
Key &key, Element **lastFound )
const
{
const
Element *lower, *mid, *upper;
long
keyRelation;
const
long
tblLen = BaseTable::length();
if
( BaseTable::data == 0 )
return
0;
lower = BaseTable::data;
upper = BaseTable::data + tblLen - 1;
while
(
true
) {
if
( upper < lower ) {
if
( lastFound != 0 )
*lastFound = (Element*)lower;
return
0;
}
mid = lower + ((upper-lower)>>1);
keyRelation =
this
->compare(key, GET_KEY(*mid));
if
( keyRelation < 0 )
upper = mid - 1;
else
if
( keyRelation > 0 )
lower = mid + 1;
else
{
if
( lastFound != 0 )
*lastFound = (Element*)mid;
return
(Element*)mid;
}
}
}
template
<BST_TEMPL_DEF> Element *BstTable<BST_TEMPL_USE>::
insert(
const
Key &key, Element **lastFound)
{
const
Element *lower, *mid, *upper;
long
keyRelation, insertPos;
const
long
tblLen = BaseTable::length();
if
( tblLen == 0 ) {
lower = BaseTable::data;
goto
insert;
}
lower = BaseTable::data;
upper = BaseTable::data + tblLen - 1;
while
(
true
) {
if
( upper < lower ) {
goto
insert;
}
mid = lower + ((upper-lower)>>1);
keyRelation =
this
->compare(key, GET_KEY(*mid));
if
( keyRelation < 0 )
upper = mid - 1;
else
if
( keyRelation > 0 )
lower = mid + 1;
else
{
if
( lastFound != 0 )
*lastFound = (Element*)mid;
return
0;
}
}
insert:
insertPos = lower - BaseTable::data;
BaseVector::makeRawSpaceFor(insertPos, 1);
new
(BaseTable::data + insertPos) Element(key);
if
( lastFound != 0 )
*lastFound = BaseTable::data + insertPos;
return
BaseTable::data + insertPos;
}
template
<BST_TEMPL_DEF> Element *BstTable<BST_TEMPL_USE>::
insertMulti(
const
Key &key)
{
const
Element *lower, *mid, *upper;
long
keyRelation, insertPos;
const
long
tblLen = BaseTable::length();
if
( tblLen == 0 ) {
lower = BaseTable::data;
goto
insert;
}
lower = BaseTable::data;
upper = BaseTable::data + tblLen - 1;
while
(
true
) {
if
( upper < lower ) {
goto
insert;
}
mid = lower + ((upper-lower)>>1);
keyRelation = compare(key, GET_KEY(*mid));
if
( keyRelation < 0 )
upper = mid - 1;
else
if
( keyRelation > 0 )
lower = mid + 1;
else
{
lower = mid;
goto
insert;
}
}
insert:
insertPos = lower - BaseTable::data;
BaseVector::makeRawSpaceFor(insertPos, 1);
new
(BaseTable::data + insertPos) Element(key);
return
BaseTable::data + insertPos;
}
template
<BST_TEMPL_DEF>
bool
BstTable<BST_TEMPL_USE>::
insert(
const
BstTable &other)
{
bool
allSuccess =
true
;
long
otherLen = other.length();
for
(
long
i = 0; i < otherLen; i++ ) {
Element *el = insert( other.data[i] );
if
( el == 0 )
allSuccess =
false
;
}
return
allSuccess;
}
template
<BST_TEMPL_DEF>
void
BstTable<BST_TEMPL_USE>::
insertMulti(
const
BstTable &other)
{
long
otherLen = other.length();
for
(
long
i = 0; i < otherLen; i++ )
insertMulti( other.data[i] );
}
#if ! defined( BSTSET )
template
<BST_TEMPL_DEF> Element *BstTable<BST_TEMPL_USE>::
insert(
const
Element &el, Element **lastFound )
{
const
Element *lower, *mid, *upper;
long
keyRelation, insertPos;
const
long
tblLen = BaseTable::length();
if
( tblLen == 0 ) {
lower = BaseTable::data;
goto
insert;
}
lower = BaseTable::data;
upper = BaseTable::data + tblLen - 1;
while
(
true
) {
if
( upper < lower ) {
goto
insert;
}
mid = lower + ((upper-lower)>>1);
keyRelation = compare(GET_KEY(el), GET_KEY(*mid));
if
( keyRelation < 0 )
upper = mid - 1;
else
if
( keyRelation > 0 )
lower = mid + 1;
else
{
if
( lastFound != 0 )
*lastFound = (Element*)mid;
return
0;
}
}
insert:
insertPos = lower - BaseTable::data;
BaseVector::makeRawSpaceFor(insertPos, 1);
new
(BaseTable::data + insertPos) Element(el);
if
( lastFound != 0 )
*lastFound = BaseTable::data + insertPos;
return
BaseTable::data + insertPos;
}
template
<BST_TEMPL_DEF> Element *BstTable<BST_TEMPL_USE>::
insertMulti(
const
Element &el)
{
const
Element *lower, *mid, *upper;
long
keyRelation, insertPos;
const
long
tblLen = BaseTable::length();
if
( tblLen == 0 ) {
lower = BaseTable::data;
goto
insert;
}
lower = BaseTable::data;
upper = BaseTable::data + tblLen - 1;
while
(
true
) {
if
( upper < lower ) {
goto
insert;
}
mid = lower + ((upper-lower)>>1);
keyRelation =
this
->compare(GET_KEY(el), GET_KEY(*mid));
if
( keyRelation < 0 )
upper = mid - 1;
else
if
( keyRelation > 0 )
lower = mid + 1;
else
{
lower = mid;
goto
insert;
}
}
insert:
insertPos = lower - BaseTable::data;
BaseVector::makeRawSpaceFor(insertPos, 1);
new
(BaseTable::data + insertPos) Element(el);
return
BaseTable::data + insertPos;
}
#endif
#if defined( BSTMAP )
template
<BST_TEMPL_DEF> Element *BstTable<BST_TEMPL_USE>::
insert(
const
Key &key,
const
Value &val, Element **lastFound)
{
const
Element *lower, *mid, *upper;
long
keyRelation, insertPos;
const
long
tblLen = BaseTable::length();
if
( tblLen == 0 ) {
lower = BaseTable::data;
goto
insert;
}
lower = BaseTable::data;
upper = BaseTable::data + tblLen - 1;
while
(
true
) {
if
( upper < lower ) {
goto
insert;
}
mid = lower + ((upper-lower)>>1);
keyRelation = Compare::compare(key, mid->key);
if
( keyRelation < 0 )
upper = mid - 1;
else
if
( keyRelation > 0 )
lower = mid + 1;
else
{
if
( lastFound != NULL )
*lastFound = (Element*)mid;
return
0;
}
}
insert:
insertPos = lower - BaseTable::data;
BaseVector::makeRawSpaceFor(insertPos, 1);
new
(BaseTable::data + insertPos) Element(key, val);
if
( lastFound != NULL )
*lastFound = BaseTable::data + insertPos;
return
BaseTable::data + insertPos;
}
template
<BST_TEMPL_DEF> Element *BstTable<BST_TEMPL_USE>::
insertMulti(
const
Key &key,
const
Value &val)
{
const
Element *lower, *mid, *upper;
long
keyRelation, insertPos;
const
long
tblLen = BaseTable::length();
if
( tblLen == 0 ) {
lower = BaseTable::data;
goto
insert;
}
lower = BaseTable::data;
upper = BaseTable::data + tblLen - 1;
while
(
true
) {
if
( upper < lower ) {
goto
insert;
}
mid = lower + ((upper-lower)>>1);
keyRelation = Compare::compare(key, mid->key);
if
( keyRelation < 0 )
upper = mid - 1;
else
if
( keyRelation > 0 )
lower = mid + 1;
else
{
lower = mid;
goto
insert;
}
}
insert:
insertPos = lower - BaseTable::data;
BaseVector::makeRawSpaceFor(insertPos, 1);
new
(BaseTable::data + insertPos) Element(key, val);
return
BaseTable::data + insertPos;
}
#endif
#ifdef AAPL_NAMESPACE
}
#endif