368 lines
17 KiB
C++
368 lines
17 KiB
C++
/*! \file picontainers.h
|
|
* \brief Generic containers
|
|
*
|
|
* This file declare all containers and useful macros
|
|
* to use them
|
|
*/
|
|
/*
|
|
PIP - Platform Independent Primitives
|
|
Generic containers
|
|
Copyright (C) 2013 Ivan Pelipenko peri4ko@gmail.com
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program 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 General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#ifndef PICONTAINERS_H
|
|
#define PICONTAINERS_H
|
|
|
|
#include "pivector.h"
|
|
#include "pistack.h"
|
|
#include "piqueue.h"
|
|
|
|
#ifdef DOXYGEN
|
|
/*! \def piForeach(i,c)
|
|
* \brief Macro for iterate any container
|
|
* \details Use this macros instead of standard "for"
|
|
* to get read/write access to each element of container.
|
|
* Pass direction is direct \n
|
|
* Example: \snippet picontainers.cpp foreach
|
|
*/
|
|
/*! \def piForeachC(i,c)
|
|
* \brief Macro for iterate any container only for read
|
|
* \details Use this macros instead of standard "for"
|
|
* to get read access to each element of container.
|
|
* Pass direction is direct \n
|
|
* Example: \snippet picontainers.cpp foreachC
|
|
*/
|
|
/*! \def piForeachR(i,c)
|
|
* \brief Macro for iterate any container with reverse direction
|
|
* \details Use this macros instead of standard "for"
|
|
* to get read/write access to each element of container.
|
|
* Pass direction is reverse \n
|
|
* Example: \snippet picontainers.cpp foreachR
|
|
*/
|
|
/*! \def piForeachCR(i,c)
|
|
* \brief Macro for iterate any container only for read with reverse direction
|
|
* \details Use this macros instead of standard "for"
|
|
* to get read access to each element of container.
|
|
* Pass direction is reverse \n
|
|
* Example: \snippet picontainers.cpp foreachCR
|
|
*/
|
|
#endif
|
|
|
|
#ifdef CC_GCC
|
|
|
|
template<typename Type>
|
|
class _PIForeach {
|
|
public:
|
|
_PIForeach(Type & t): _t(t) {_it = _t.begin(); _break = false;}
|
|
typename Type::value_type _var;
|
|
typename Type::iterator _it;
|
|
Type & _t;
|
|
bool _break;
|
|
inline bool isEnd() {return _it == _t.end();}
|
|
inline void operator ++() {_it++; _break = false;}
|
|
};
|
|
|
|
template<typename Type>
|
|
class _PIForeachR {
|
|
public:
|
|
_PIForeachR(Type & t): _t(t) {_rit = _t.rbegin(); _break = false;}
|
|
typename Type::value_type _var;
|
|
typename Type::reverse_iterator _rit;
|
|
Type & _t;
|
|
bool _break;
|
|
inline bool isEnd() {return _rit == _t.rend();}
|
|
inline void operator ++() {_rit++; _break = false;}
|
|
};
|
|
|
|
template<typename Type>
|
|
class _PIForeachC {
|
|
public:
|
|
_PIForeachC(const Type & t): _t(t) {_it = _t.begin(); _break = false;}
|
|
typename Type::value_type _var;
|
|
typename Type::const_iterator _it;
|
|
const Type & _t;
|
|
bool _break;
|
|
inline bool isEnd() {return _it == _t.end();}
|
|
inline void operator ++() {_it++; _break = false;}
|
|
};
|
|
|
|
template<typename Type>
|
|
class _PIForeachCR {
|
|
public:
|
|
_PIForeachCR(const Type & t): _t(t) {_rit = _t.rbegin(); _break = false;}
|
|
typename Type::value_type _var;
|
|
typename Type::const_reverse_iterator _rit;
|
|
const Type & _t;
|
|
bool _break;
|
|
inline bool isEnd() {return _rit == _t.rend();}
|
|
inline void operator ++() {_rit++; _break = false;}
|
|
};
|
|
|
|
#define piForeach(i,c) for(_PIForeach<typeof(c)> _for(c); !_for.isEnd(); ++_for) \
|
|
for(i(*_for._it); !_for._break; _for._break = true)
|
|
#define piForeachR(i,c) for(_PIForeachR<typeof(c)> _for(c); !_for.isEnd(); ++_for) \
|
|
for(i(*_for._rit); !_for._break; _for._break = true)
|
|
#define piForeachA(i,c) for(_PIForeach<typeof(c)> _for(c); !_for.isEnd(); ++_for) \
|
|
for(typeof(_for._var) & i(*_for._it); !_for._break; _for._break = true)
|
|
#define piForeachAR(i,c) for(_PIForeachR<typeof(c)> _for(c); !_for.isEnd(); ++_for) \
|
|
for(typeof(_for._var) & i(*_for._rit); !_for._break; _for._break = true)
|
|
#define piForeachC(i,c) for(_PIForeachC<typeof(c)> _for(c); !_for.isEnd(); ++_for) \
|
|
for(const i(*_for._it); !_for._break; _for._break = true)
|
|
#define piForeachCR(i,c) for(_PIForeachCR<typeof(c)> _for(c); !_for.isEnd(); ++_for) \
|
|
for(const i(*_for._rit); !_for._break; _for._break = true)
|
|
#define piForeachCA(i,c) for(_PIForeachC<typeof(c)> _for(c); !_for.isEnd(); ++_for) \
|
|
for(const typeof(_for._var) & i(*_for._it); !_for._break; _for._break = true)
|
|
#define piForeachCAR(i,c) for(_PIForeachCR<typeof(c)> _for(c); !_for.isEnd(); ++_for) \
|
|
for(const typeof(_for._var) & i(*_for._rit); !_for._break; _for._break = true)
|
|
|
|
#define piForeachRA piForeachAR
|
|
#define piForeachAC piForeachCA
|
|
#define piForeachCRA piForeachCAR
|
|
#define piForeachARC piForeachCAR
|
|
#define piForeachACR piForeachCAR
|
|
#define piForeachRCA piForeachCAR
|
|
#define piForeachRAC piForeachCAR
|
|
|
|
#else
|
|
|
|
struct _PIForeachBase {mutable bool _break;};
|
|
|
|
template<typename Type>
|
|
class _PIForeach: public _PIForeachBase {
|
|
public:
|
|
_PIForeach(Type & t, bool i = false): _t(t), _inv(i) {if (_inv) _rit = _t.rbegin(); else _it = _t.begin(); _break = false;}
|
|
mutable typename Type::value_type _var;
|
|
mutable typename Type::iterator _it;
|
|
mutable typename Type::reverse_iterator _rit;
|
|
Type & _t;
|
|
bool _inv;
|
|
bool isEnd() {if (_inv) return _rit == _t.rend(); else return _it == _t.end();}
|
|
void operator ++() {if (_inv) _rit++; else _it++; _break = false;}
|
|
};
|
|
|
|
template<typename Type>
|
|
class _PIForeachC: public _PIForeachBase {
|
|
public:
|
|
_PIForeachC(const Type & t, bool i = false): _t(t), _inv(i) {if (_inv) _rit = _t.rbegin(); else _it = _t.begin(); _break = false;}
|
|
mutable typename Type::value_type _var;
|
|
mutable typename Type::const_iterator _it;
|
|
mutable typename Type::const_reverse_iterator _rit;
|
|
const Type & _t;
|
|
bool _inv;
|
|
bool isEnd() {if (_inv) return _rit == _t.rend(); else return _it == _t.end();}
|
|
void operator ++() {if (_inv) _rit++; else _it++; _break = false;}
|
|
};
|
|
|
|
template <typename T> inline _PIForeach<T> _PIForeachNew(T & t, bool i = false) {return _PIForeach<T>(t, i);}
|
|
template <typename T> inline _PIForeach<T> * _PIForeachCast(_PIForeachBase & c, T & ) {return static_cast<_PIForeach<T> * >(&c);}
|
|
|
|
template <typename T> inline _PIForeachC<T> _PIForeachNewC(const T & t, bool i = false) {return _PIForeachC<T>(t, i);}
|
|
template <typename T> inline _PIForeachC<T> * _PIForeachCastC(_PIForeachBase & c, const T & ) {return static_cast<_PIForeachC<T> * >(&c);}
|
|
|
|
#define piForeach(i,c) for(_PIForeachBase & _for = _PIForeachNew(c); !_PIForeachCast(_for, c)->isEnd(); ++(*_PIForeachCast(_for, c))) \
|
|
for(i = *(_PIForeachCast(_for, c)->_it); !_for._break; _for._break = true)
|
|
#define piForeachR(i,c) for(_PIForeachBase & _for = _PIForeachNew(c, true); !_PIForeachCast(_for, c)->isEnd(); ++(*_PIForeachCast(_for, c))) \
|
|
for(i = *(_PIForeachCast(_for, c)->_rit); !_for._break; _for._break = true)
|
|
#define piForeachC(i,c) for(_PIForeachBase & _for = _PIForeachNewC(c); !_PIForeachCastC(_for, c)->isEnd(); ++(*_PIForeachCastC(_for, c))) \
|
|
for(const i = *(_PIForeachCastC(_for, c)->_it); !_for._break; _for._break = true)
|
|
#define piForeachCR(i,c) for(_PIForeachBase & _for = _PIForeachNewC(c, false); !_PIForeachCastC(_for, c)->isEnd(); ++(*_PIForeachCastC(_for, c))) \
|
|
for(const i = *(_PIForeachCastC(_for, c)->_rit); !_for._break; _for._break = true)
|
|
|
|
#endif
|
|
|
|
#define piForeachRC piForeachCR
|
|
|
|
#define piForTimes(c) for(int _i##c = 0; _i##c < c; ++_i##c)
|
|
|
|
|
|
template<typename Type, typename Allocator = std::allocator<Type> >
|
|
class PIP_EXPORT PIList: public list<Type, Allocator> {
|
|
typedef PIList<Type, Allocator> _CList;
|
|
typedef list<Type, Allocator> _stlc;
|
|
public:
|
|
PIList() {piMonitor.containers++;}
|
|
PIList(const Type & value) {piMonitor.containers++; _stlc::resize(1, value);}
|
|
PIList(const Type & v0, const Type & v1) {piMonitor.containers++; _stlc::push_back(v0); _stlc::push_back(v1);}
|
|
PIList(const Type & v0, const Type & v1, const Type & v2) {piMonitor.containers++; _stlc::push_back(v0); _stlc::push_back(v1); _stlc::push_back(v2);}
|
|
PIList(const Type & v0, const Type & v1, const Type & v2, const Type & v3) {piMonitor.containers++; _stlc::push_back(v0); _stlc::push_back(v1); _stlc::push_back(v2); _stlc::push_back(v3);}
|
|
PIList(uint size, const Type & value = Type()) {piMonitor.containers++; _stlc::resize(size, value);}
|
|
~PIList() {piMonitor.containers--;}
|
|
Type & operator [](uint index) {return (*this)[index];}
|
|
Type & operator [](uint index) const {return (*this)[index];}
|
|
const Type * data(uint index = 0) const {return &(*this)[index];}
|
|
Type * data(uint index = 0) {return &(*this)[index];}
|
|
int size_s() const {return static_cast<int>(_stlc::size());}
|
|
bool isEmpty() const {return _stlc::empty();}
|
|
bool has(const Type & t) const {for (typename _stlc::const_iterator i = _stlc::begin(); i != _stlc::end(); ++i) if (t == *i) return true; return false;}
|
|
int etries(const Type & t) const {int ec = 0; for (typename _stlc::const_iterator i = _stlc::begin(); i != _stlc::end(); ++i) if (t == *i) ++ec; return ec;}
|
|
_CList & fill(const Type & t) {_stlc::assign(_stlc::size(), t); return *this;}
|
|
_CList & remove(uint index) {_stlc::erase(_stlc::begin() + index); return *this;}
|
|
_CList & remove(uint index, uint count) {_stlc::erase(_stlc::begin() + index, _stlc::begin() + index + count); return *this;}
|
|
_CList & insert(uint pos, const Type & t) {_stlc::insert(_stlc::begin() + pos, t); return *this;}
|
|
_CList & operator <<(const Type & t) {_stlc::push_back(t); return *this;}
|
|
PIVector<Type> toVector() {PIVector<Type> v; for (typename _stlc::const_iterator i = _stlc::begin(); i != _stlc::end(); ++i) v << *i; return v;}
|
|
};
|
|
|
|
/*! \brief Set of any type
|
|
* \details This class used to store collection of unique elements
|
|
* of any type. You can only add values to set with \a operator<< or
|
|
* with function \a insert(). You can discover if value already in
|
|
* set with \a operator[] or with function \a find(). These function
|
|
* has logarithmic complexity.
|
|
*/
|
|
template<typename Type, typename Compare = std::less<Type>, typename Allocator = std::allocator<Type> >
|
|
class PIP_EXPORT PISet: public set<Type, Compare, Allocator> {
|
|
typedef PISet<Type, Compare, Allocator> _CSet;
|
|
typedef set<Type, Compare, Allocator> _stlc;
|
|
public:
|
|
|
|
//! Contructs an empty set
|
|
PISet() {piMonitor.containers++;}
|
|
|
|
//! Contructs set with one element "value"
|
|
PISet(const Type & value) {piMonitor.containers++; _stlc::resize(1, value);}
|
|
|
|
//! Contructs set with elements "v0" and "v1"
|
|
PISet(const Type & v0, const Type & v1) {piMonitor.containers++; _stlc::insert(v0); _stlc::insert(v1);}
|
|
|
|
//! Contructs set with elements "v0", "v1" and "v2"
|
|
PISet(const Type & v0, const Type & v1, const Type & v2) {piMonitor.containers++; _stlc::insert(v0); _stlc::insert(v1); _stlc::insert(v2);}
|
|
|
|
//! Contructs set with elements "v0", "v1", "v2" and "v3"
|
|
PISet(const Type & v0, const Type & v1, const Type & v2, const Type & v3) {piMonitor.containers++; _stlc::insert(v0); _stlc::insert(v1); _stlc::insert(v2); _stlc::insert(v3);}
|
|
|
|
~PISet() {piMonitor.containers--;}
|
|
|
|
//! Returns elements count
|
|
int size_s() const {return static_cast<int>(_stlc::size());}
|
|
|
|
//! Returns if set is empty
|
|
bool isEmpty() const {return _stlc::empty();}
|
|
|
|
#ifdef DOXYGEN
|
|
|
|
//! Clear th set
|
|
void clear();
|
|
|
|
//! Insert element "t" if it doesn`t exists in this set
|
|
void insert(const Type & t);
|
|
|
|
#endif
|
|
|
|
_CSet & remove(uint index) {_stlc::erase(_stlc::begin() + index); return *this;}
|
|
_CSet & remove(uint index, uint count) {_stlc::erase(_stlc::begin() + index, _stlc::begin() + index + count); return *this;}
|
|
_CSet & operator <<(const Type & t) {_stlc::insert(t); return *this;}
|
|
|
|
//! Returns if element "t" exists in this set
|
|
bool operator [](const Type & t) {return _stlc::find(t);}
|
|
|
|
//! Returns content of set as PIVector
|
|
PIVector<Type> toVector() {PIVector<Type> v; for (typename _stlc::const_iterator i = _stlc::begin(); i != _stlc::end(); ++i) v << *i; return v;}
|
|
};
|
|
|
|
|
|
|
|
template<typename Type, typename Allocator = std::allocator<Type> >
|
|
class PIP_EXPORT PIDeque: public deque<Type, Allocator> {
|
|
typedef PIDeque<Type, Allocator> _CDeque;
|
|
typedef deque<Type, Allocator> _stlc;
|
|
public:
|
|
PIDeque() {piMonitor.containers++;}
|
|
PIDeque(const Type & value) {piMonitor.containers++; _stlc::resize(1, value);}
|
|
PIDeque(const Type & v0, const Type & v1) {piMonitor.containers++; _stlc::push_back(v0); _stlc::push_back(v1);}
|
|
PIDeque(const Type & v0, const Type & v1, const Type & v2) {piMonitor.containers++; _stlc::push_back(v0); _stlc::push_back(v1); _stlc::push_back(v2);}
|
|
PIDeque(const Type & v0, const Type & v1, const Type & v2, const Type & v3) {piMonitor.containers++; _stlc::push_back(v0); _stlc::push_back(v1); _stlc::push_back(v2); _stlc::push_back(v3);}
|
|
~PIDeque() {piMonitor.containers--;}
|
|
int size_s() const {return static_cast<int>(_stlc::size());}
|
|
bool isEmpty() const {return _stlc::empty();}
|
|
bool has(const Type & t) const {for (typename _stlc::const_iterator i = _stlc::begin(); i != _stlc::end(); ++i) if (t == *i) return true; return false;}
|
|
int etries(const Type & t) const {int ec = 0; for (typename _stlc::const_iterator i = _stlc::begin(); i != _stlc::end(); ++i) if (t == *i) ++ec; return ec;}
|
|
_CDeque & operator <<(const Type & t) {_CDeque::push_back(t); return *this;}
|
|
PIVector<Type> toVector() {PIVector<Type> v; for (typename _stlc::const_iterator i = _stlc::begin(); i != _stlc::end(); ++i) v << *i; return v;}
|
|
};
|
|
|
|
|
|
template<typename Type0, typename Type1>
|
|
class PIP_EXPORT PIPair {
|
|
public:
|
|
PIPair() {first = Type0(); second = Type1();}
|
|
PIPair(const Type0 & value0, const Type1 & value1) {first = value0; second = value1;}
|
|
Type0 first;
|
|
Type1 second;
|
|
};
|
|
template<typename Type0, typename Type1>
|
|
inline bool operator <(const PIPair<Type0, Type1> & value0, const PIPair<Type0, Type1> & value1) {return value0.first < value1.first;}
|
|
template<typename Type0, typename Type1>
|
|
inline std::ostream & operator <<(std::ostream & s, const PIPair<Type0, Type1> & v) {s << "(" << v.first << ", " << v.second << ")"; return s;}
|
|
template<typename Type0, typename Type1>
|
|
inline PICout operator <<(PICout s, const PIPair<Type0, Type1> & v) {s.space(); s.setControl(0, true); s << "(" << v.first << ", " << v.second << ")"; s.restoreControl(); return s;}
|
|
|
|
|
|
template<typename Key, typename Type>
|
|
class PIP_EXPORT PIMap: public map<Key, Type> {
|
|
typedef PIMap<Key, Type> _CMap;
|
|
typedef map<Key, Type> _stlc;
|
|
typedef std::pair<Key, Type> _stlpair;
|
|
public:
|
|
PIMap() {;}
|
|
PIMap(const Key & key_, const Type & value_) {insert(key_, value_);}
|
|
bool isEmpty() const {return _stlc::empty();}
|
|
bool contains(const Key & key_) const {return _stlc::count(key_) > 0;}
|
|
_CMap & insert(const Key & key_, const Type & value_) {_stlc::insert(_stlpair(key_, value_)); return *this;}
|
|
_CMap & insert(PIPair<Key, Type> entry_) {_stlc::insert(_stlpair(entry_.first, entry_.second)); return *this;}
|
|
Key key(Type value_) const {for (typename _stlc::const_iterator i = _stlc::begin(); i != _stlc::end(); i++) if (i->second == value_) return i->first; return Key();}
|
|
Type & value(const Key & key_) {typename _stlc::iterator it = _stlc::find(key_); if (it == _stlc::end()) it->second = Type(); return it->second;}
|
|
Type & at(const Key & key_) {return value(key_);}
|
|
Type value(const Key & key_) const {return _stlc::find(key_)->second;}
|
|
};
|
|
|
|
|
|
template<typename Key, typename Type>
|
|
class PIP_EXPORT PIMultiMap: public multimap<Key, Type> {
|
|
typedef PIMultiMap<Key, Type> _CMultiMap;
|
|
typedef multimap<Key, Type> _stlc;
|
|
typedef std::pair<Key, Type> _stlpair;
|
|
public:
|
|
PIMultiMap() {;}
|
|
PIMultiMap(const Key & key_, const Type & value_) {insert(key_, value_);}
|
|
_CMultiMap & insert(const Key & key_, const Type & value_) {_stlc::insert(_stlpair(key_, value_)); return *this;}
|
|
_CMultiMap & insert(PIPair<Key, Type> entry_) {_stlc::insert(_stlpair(entry_.first, entry_.second)); return *this;}
|
|
bool isEmpty() const {return _stlc::empty();}
|
|
bool contains(const Key & key_) const {return _stlc::count(key_) > 0;}
|
|
Key key(Type value_) const {for (typename _stlc::const_iterator i = _stlc::begin(); i != _stlc::end(); i++) if (i->second == value_) return i->first; return Key();}
|
|
PIVector<Key> keys(Type value_) const {
|
|
PIVector<Key> ret;
|
|
for (typename _stlc::const_iterator i = _stlc::begin(); i != _stlc::end(); i++)
|
|
if (i->second == value_)
|
|
ret << i->first;
|
|
return ret;
|
|
}
|
|
Type & value(const Key & key_) {return _stlc::find(key_)->second;}
|
|
Type value(const Key & key_) const {return _stlc::find(key_)->second;}
|
|
PIVector<Type> values(const Key & key_) const {
|
|
std::pair<typename _stlc::const_iterator, typename _stlc::const_iterator> range = _stlc::equal_range(key_);
|
|
PIVector<Type> ret;
|
|
for (typename _stlc::const_iterator i = range.first; i != range.second; ++i)
|
|
ret << i->second;
|
|
return ret;
|
|
}
|
|
Type & operator [](const Key & key_) {if (!contains(key_)) return _stlc::insert(_stlpair(key_, Type()))->second; return _stlc::find(key_)->second;}
|
|
Type operator [](const Key & key_) const {return _stlc::find(key_)->second;}
|
|
};
|
|
|
|
|
|
#endif // PICONTAINERS_H
|