LUMIERA.clone/src/lib/scoped-ptrvect.hpp

182 lines
4.9 KiB
C++
Raw Normal View History

/*
SCOPED-PTRVECT.hpp - simple noncopyable lifecycle managing collection of pointers
Copyright (C) Lumiera.org
2009, Hermann Vosseler <Ichthyostega@web.de>
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 2 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, write to the Free Software
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
*/
/** @file scoped-ptrvect.hpp
** Managing lifecycle for a collection of objects. Sometimes we need to
** build and own a number of objects, including lifecycle management.
** For example, a service provider may need to maintain a number of individual
** process handles. The solution here is deliberately kept simple, it is
** similar to using a stl container with shared_ptr(s), but behaves rather
** like boost::scoped_ptr. It provides the same basic functionality as
** boost::ptr_vector, but doesn't require us to depend on boost-serialisation.
**
** Some details to note:
** - contained objects accessed by reference, never NULL.
** - TODO: iterators, detaching of objects...
** - TODO: retro-fit with refarray interface (--> builder branch)
**
** @see scoped-ptrvect-test.cpp
2009-07-04 00:24:55 +02:00
** @see scoped-holder.hpp
** @see gui::DisplayService usage example
*/
#ifndef LIB_SCOPED_PTRVECT_H
#define LIB_SCOPED_PTRVECT_H
#include "include/logging.h"
#include "lib/error.hpp"
#include "lib/util.hpp"
#include <vector>
#include <boost/noncopyable.hpp>
namespace lib {
using util::for_each;
/**
* Simple vector based collection of pointers, noncopyable and managing
* lifecycle of the pointed-to objects. Implemented by a vector of
* bare pointers (private inheritance)
*/
template<class T>
class ScopedPtrVect
: std::vector<T*>,
boost::noncopyable
{
typedef std::vector<T*> _Vec;
public:
typedef size_t size_type;
typedef T & reference;
typedef T const& const_reference;
ScopedPtrVect ()
: _Vec()
{ }
explicit
ScopedPtrVect (size_type capacity)
: _Vec()
{
_Vec::reserve (capacity);
}
~ScopedPtrVect ()
{
clear();
}
/** take ownership of the given object,
* adding it at the end of the collection
* @note object is deleted in case of any
* problem while adding it
*/
T&
manage (T* obj)
{
if (obj)
try
{
push_back (obj);
return *obj;
}
catch(...)
{
delete obj;
throw;
} }
void
clear()
{
typedef typename _Vec::iterator VIter;
VIter e = this->end();
for (VIter i = this->begin(); i!=e; ++i)
{
if (*i)
try
{
delete *i;
*i = 0;
}
catch(std::exception& ex)
{
WARN (library, "Problem while deallocating ScopedPtrVect: %s", ex.what());
} }
_Vec::clear();
}
2009-07-11 19:23:20 +02:00
/* === Element access and iteration === */
T&
operator[] (size_type i)
{
return *get(i);
}
typedef PtrDerefIter<reference, _Vec> iterator;
typedef PtrDerefIter<const_reference, _Vec> const_iterator;
iterator begin() { return iterator (_Vec::begin()); }
const_iterator begin() const { return const_iterator (_Vec::begin()); }
iterator end() { return iterator (); }
const_iterator end() const { return const_iterator (); }
/* ====== proxied vector functions ==================== */
size_type size () const { return _Vec::size(); }
size_type max_size () const { return _Vec::max_size(); }
size_type capacity () const { return _Vec::capacity(); }
bool empty () const { return _Vec::empty(); }
private:
2009-07-11 19:23:20 +02:00
/** internal element access, including null check */
T* get(size_type i)
{
T* p (_Vec::at (i));
if (!p)
throw lumiera::error::Invalid("no valid object at this index");
else
return p;
}
};
} // namespace lib
#endif