mirror of
https://git.lyx.org/repos/lyx.git
synced 2025-01-04 08:37:52 +00:00
338 lines
12 KiB
C
338 lines
12 KiB
C
|
////////////////////////////////////////////////////////////////////////////////
|
||
|
//
|
||
|
// Visual Leak Detector - Lightweight STL-like Set Template
|
||
|
// Copyright (c) 2006 Dan Moulding
|
||
|
//
|
||
|
// This library is free software; you can redistribute it and/or
|
||
|
// modify it under the terms of the GNU Lesser General Public
|
||
|
// License as published by the Free Software Foundation; either
|
||
|
// version 2.1 of the License, or (at your option) any later version.
|
||
|
//
|
||
|
// This library is distributed in the hope that it will be useful,
|
||
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||
|
// Lesser General Public License for more details.
|
||
|
//
|
||
|
// You should have received a copy of the GNU Lesser General Public
|
||
|
// License along with this library; if not, write to the Free Software
|
||
|
// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||
|
//
|
||
|
// See COPYING.txt for the full terms of the GNU Lesser General Public License.
|
||
|
//
|
||
|
////////////////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
#pragma once
|
||
|
|
||
|
#ifndef VLDBUILD
|
||
|
#error \
|
||
|
"This header should only be included by Visual Leak Detector when building it from source. \
|
||
|
Applications should never include this header."
|
||
|
#endif
|
||
|
|
||
|
#include "tree.h" // Provides access to the Tree template class.
|
||
|
|
||
|
////////////////////////////////////////////////////////////////////////////////
|
||
|
//
|
||
|
// The Set Template Class
|
||
|
//
|
||
|
// This is a lightweight STL-like set template. It makes use of the Tree class
|
||
|
// template to enable fast insert, find, and erase operations.
|
||
|
//
|
||
|
// Note that while this is a STL-like class, it is not a full STL-compliant
|
||
|
// implementation of the STL set container. It contains just the bare minimum
|
||
|
// functionality required by Visual Leak Detector. Because of its "lightweight"
|
||
|
// nature, this set class has a noticeable performance advantage over some
|
||
|
// other standard STL set implementations.
|
||
|
//
|
||
|
template <typename Tk>
|
||
|
class Set {
|
||
|
public:
|
||
|
class Iterator {
|
||
|
public:
|
||
|
// Constructor
|
||
|
Iterator ()
|
||
|
{
|
||
|
// Plainly constructed iterators don't reference anything.
|
||
|
m_node = NULL;
|
||
|
m_tree = NULL;
|
||
|
}
|
||
|
|
||
|
// operator != - Inequality operator for Set Iterators. Two Set
|
||
|
// Iterators are considered equal if and only if they both reference
|
||
|
// the same key in the same Set.
|
||
|
//
|
||
|
// - other (IN): The other Set Iterator to compare against.
|
||
|
//
|
||
|
// Return Value:
|
||
|
//
|
||
|
// Returns true if the specified Set Iterator is not equal to this
|
||
|
// Set Iterator; otherwise, returns false.
|
||
|
//
|
||
|
BOOL operator != (const Iterator &other) const
|
||
|
{
|
||
|
return ((m_tree != other.m_tree) || (m_node != other.m_node));
|
||
|
}
|
||
|
|
||
|
// operator * - Dereference operator for Set Iterators.
|
||
|
//
|
||
|
// Note: The reference returned by this function is "const", so the
|
||
|
// value referenced by the Iterator may not be modified through the
|
||
|
// Iterator. This is a departure from STL iterator behavior.
|
||
|
//
|
||
|
// Also, dereferencing an Iterator which does not reference a valid
|
||
|
// value in the Set is undefined and will almost certainly cause a
|
||
|
// crash.
|
||
|
//
|
||
|
// Return Value:
|
||
|
//
|
||
|
// Returns a const reference to the key in the Map referenced by the
|
||
|
// Iterator.
|
||
|
//
|
||
|
const Tk& operator * () const
|
||
|
{
|
||
|
return m_node->key;
|
||
|
}
|
||
|
|
||
|
// operator ++ - Prefix increment operator for Set Iterators. Causes the
|
||
|
// Iterator to reference the in-oder successor of the key currently
|
||
|
// referenced by the Iterator. If the Iterator is currently
|
||
|
// referencing the largest key in the Map, then the resulting Iterator
|
||
|
// will reference the Set's end (the NULL pair).
|
||
|
//
|
||
|
// Note: Incrementing an Iterator which does not reference a valid
|
||
|
// key in the Set is undefined and will almost certainly cause a
|
||
|
// crash.
|
||
|
//
|
||
|
// Return Value:
|
||
|
//
|
||
|
// Returns the Iterator after it has been incremented.
|
||
|
//
|
||
|
Iterator& operator ++ (int)
|
||
|
{
|
||
|
m_node = m_tree->next(m_node);
|
||
|
return *this;
|
||
|
}
|
||
|
|
||
|
// operator ++ - Postfix increment operator for Map Iterators. Causes
|
||
|
// the Iterator to reference the in-order successor of the key/value
|
||
|
// pair currently referenced by the Iterator. If the Iterator is
|
||
|
// currently referencing the largest key/value pair in the Map, then
|
||
|
// the resulting Iterator will reference the Map's end (the NULL
|
||
|
// pair).
|
||
|
//
|
||
|
// Note: Incrementing an Iterator which does not reference a valid
|
||
|
// key/value pair in the Map is undefined and will almost certainly
|
||
|
// cause a crash.
|
||
|
//
|
||
|
// Return Value:
|
||
|
//
|
||
|
// Returns the Iterator before it has been incremented.
|
||
|
//
|
||
|
Iterator operator ++ ()
|
||
|
{
|
||
|
typename Tree<Tk>::node_t *cur = m_node;
|
||
|
|
||
|
m_node = m_tree->next(m_node);
|
||
|
return Iterator(m_tree, cur);
|
||
|
}
|
||
|
|
||
|
// operator - - Subtraction operator for Set Iterators. Causes the
|
||
|
// the Iterator to reference a key that is an in-order predecessor of
|
||
|
// the currently refereced key.
|
||
|
//
|
||
|
// - num (IN): Number indicating the number of preceding keys to
|
||
|
// decrement the iterator.
|
||
|
//
|
||
|
// Return Value:
|
||
|
//
|
||
|
// Returns an Iterator referencing the key that precedes the original
|
||
|
// Iterator by "num" keys.
|
||
|
//
|
||
|
Iterator operator - (SIZE_T num) const
|
||
|
{
|
||
|
SIZE_T count;
|
||
|
typename Tree<Tk>::node_t *cur = m_node;
|
||
|
|
||
|
cur = m_tree->prev(m_node);
|
||
|
for (count = 0; count < num; count++) {
|
||
|
cur = m_tree->prev(m_node);
|
||
|
if (cur == NULL) {
|
||
|
return Iterator(m_tree, NULL);
|
||
|
}
|
||
|
}
|
||
|
return Iterator(m_tree, cur);
|
||
|
}
|
||
|
|
||
|
// operator == - Equality operator for Set Iterators. Set Iterators are
|
||
|
// considered equal if and only if they both refernce the same
|
||
|
// key in the same Set.
|
||
|
//
|
||
|
// - other (IN): The other Set Iterator to compare against.
|
||
|
//
|
||
|
// Return Value:
|
||
|
//
|
||
|
// Returns true if the specified Set Iterator is equal to this Set
|
||
|
// Iterator; otherwise returns false.
|
||
|
//
|
||
|
BOOL operator == (const Iterator &other) const
|
||
|
{
|
||
|
return ((m_tree == other.m_tree) && (m_node == other.m_node));
|
||
|
}
|
||
|
|
||
|
private:
|
||
|
// Private constructor. Only the Set class itself may use this
|
||
|
// constructor. It is used for constructing Iterators which reference
|
||
|
// specific nodes in the internal tree's structure.
|
||
|
Iterator (const Tree<Tk> *tree, typename Tree<Tk>::node_t *node)
|
||
|
{
|
||
|
m_node = node;
|
||
|
m_tree = tree;
|
||
|
}
|
||
|
|
||
|
protected:
|
||
|
typename Tree<Tk>::node_t *m_node; // Pointer to the node referenced by the Set Iterator.
|
||
|
const Tree<Tk> *m_tree; // Pointer to the tree containing the referenced node.
|
||
|
|
||
|
// The Set class is a friend of Set Iterators.
|
||
|
friend class Set<Tk>;
|
||
|
};
|
||
|
|
||
|
// Muterator class - This class provides a mutable Iterator (the regular
|
||
|
// Iterators are const Iterators). By dereferencing a Muterator, you get
|
||
|
// a modifiable element.
|
||
|
//
|
||
|
// Caution: Modifing an element in a way that changes its sorting value
|
||
|
// will corrupt the Set container. Muterators should only be used when
|
||
|
// you are absolutely certain you will not be using it to make a
|
||
|
// modification that changes the sort order of the referenced element.
|
||
|
//
|
||
|
class Muterator : public Iterator
|
||
|
{
|
||
|
public:
|
||
|
// operator = - Assignment operator for Set Muterators. Can be used to
|
||
|
// copy a Muterator from an existing Iterator, such that the Muterator
|
||
|
// references the same element referenced by the Iterator.
|
||
|
Muterator& operator = (const Iterator& other) {
|
||
|
*(Iterator*)this = other;
|
||
|
return *this;
|
||
|
}
|
||
|
|
||
|
// operator * - Dereference operator for Set Muterators.
|
||
|
//
|
||
|
// Note: Dereferencing a Muterator which does not reference a valid
|
||
|
// value in the Set is undefined and will almost certainly cause a
|
||
|
// crash.
|
||
|
//
|
||
|
// Return Value:
|
||
|
//
|
||
|
// Returns a reference to the key in the Map referenced by the
|
||
|
// Muterator.
|
||
|
//
|
||
|
Tk& operator * ()
|
||
|
{
|
||
|
return m_node->key;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
// begin - Obtains an Iterator referencing the beginning of the Set (i.e.
|
||
|
// the lowest key currently stored in the Set).
|
||
|
//
|
||
|
// Return Value:
|
||
|
//
|
||
|
// Returns an Iterator referencing the first key in the Set. If no keys
|
||
|
// are currenly stored in the Set, returns the "NULL" Iterator.
|
||
|
//
|
||
|
Iterator begin () const
|
||
|
{
|
||
|
return Iterator(&m_tree, m_tree.begin());
|
||
|
}
|
||
|
|
||
|
// end - Obtains an Iterator referencing the end of the Set. The end of
|
||
|
// the Set does not reference an actual key. Instead it represents a
|
||
|
// "null" key which signifies the end (i.e. just beyond largest key
|
||
|
// currently stored in the Set). Also known as the "NULL" Iterator.
|
||
|
//
|
||
|
// Return Value:
|
||
|
//
|
||
|
// Returns the "NULL" Iterator, signifying the end of the Set.
|
||
|
//
|
||
|
Iterator end () const
|
||
|
{
|
||
|
return Iterator(&m_tree, NULL);
|
||
|
}
|
||
|
|
||
|
// erase - Erases a key from the Set.
|
||
|
//
|
||
|
// - it (IN): Iterator referencing the key to be erased from the Set.
|
||
|
//
|
||
|
// Return Value:
|
||
|
//
|
||
|
// None.
|
||
|
//
|
||
|
VOID erase (Iterator& it)
|
||
|
{
|
||
|
m_tree.erase(it.m_node);
|
||
|
}
|
||
|
|
||
|
// erase - Erases a key from the Set.
|
||
|
//
|
||
|
// - key (IN): The key to be erased from the Set.
|
||
|
//
|
||
|
// Return Value:
|
||
|
//
|
||
|
// None.
|
||
|
//
|
||
|
VOID erase (const Tk &key)
|
||
|
{
|
||
|
m_tree.erase(key);
|
||
|
}
|
||
|
|
||
|
// find - Finds a key in the Set.
|
||
|
//
|
||
|
// - key (IN): The key to be found.
|
||
|
//
|
||
|
// Return Value:
|
||
|
//
|
||
|
// Returns an Iterator referencing the found key. If the key could not
|
||
|
// be found, then the "NULL" Iterator is returned.
|
||
|
//
|
||
|
Iterator find (const Tk &key) const
|
||
|
{
|
||
|
return Iterator(&m_tree, m_tree.find(key));
|
||
|
}
|
||
|
|
||
|
// insert - Inserts a key into the Set.
|
||
|
//
|
||
|
// - key (IN): The key to be inserted.
|
||
|
//
|
||
|
// Return Value:
|
||
|
//
|
||
|
// Returns an Iterator referencing the key after it has been inserted
|
||
|
// into the Set. If an element with an identical key value already exists
|
||
|
// in the Set, then the NULL Iterator is returned.
|
||
|
//
|
||
|
Iterator insert (const Tk &key)
|
||
|
{
|
||
|
return Iterator(&m_tree, m_tree.insert(key));
|
||
|
}
|
||
|
|
||
|
// reserve - Sets the reserve size of the Set. The reserve size is the
|
||
|
// number of keys for which space should be pre-allocated to avoid
|
||
|
// frequent heap hits when inserting new keys into the Set.
|
||
|
//
|
||
|
// - count (IN): The number of keys for which to reserve space in advance.
|
||
|
//
|
||
|
// Return Value:
|
||
|
//
|
||
|
// Returns the reserve size previously in use by the Set.
|
||
|
//
|
||
|
size_t reserve (size_t count)
|
||
|
{
|
||
|
return m_tree.reserve(count);
|
||
|
}
|
||
|
|
||
|
private:
|
||
|
// Private data
|
||
|
Tree<Tk> m_tree; // The keys are actually stored in a tree.
|
||
|
};
|