speech-tools/include/EST_THandle.h

153 lines
5.7 KiB
C
Raw Normal View History

2015-09-19 10:52:26 +02:00
/************************************************************************/
/* */
/* Centre for Speech Technology Research */
/* University of Edinburgh, UK */
/* Copyright (c) 1996,1997 */
/* All Rights Reserved. */
/* */
/* Permission is hereby granted, free of charge, to use and distribute */
/* this software and its documentation without restriction, including */
/* without limitation the rights to use, copy, modify, merge, publish, */
/* distribute, sublicense, and/or sell copies of this work, and to */
/* permit persons to whom this work is furnished to do so, subject to */
/* the following conditions: */
/* 1. The code must retain the above copyright notice, this list of */
/* conditions and the following disclaimer. */
/* 2. Any modifications must be clearly marked as such. */
/* 3. Original authors' names are not deleted. */
/* 4. The authors' names are not used to endorse or promote products */
/* derived from this software without specific prior written */
/* permission. */
/* */
/* THE UNIVERSITY OF EDINBURGH AND THE CONTRIBUTORS TO THIS WORK */
/* DISCLAIM ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING */
/* ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT */
/* SHALL THE UNIVERSITY OF EDINBURGH NOR THE CONTRIBUTORS BE LIABLE */
/* FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES */
/* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN */
/* AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, */
/* ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF */
/* THIS SOFTWARE. */
/* */
/*************************************************************************/
#ifndef __EST_THANDLE_H__
#define __EST_THANDLE_H__
#include <iostream>
using namespace std;
#include "EST_bool.h"
/** A `smart' pointer which does reference counting.
*
* Behaves almost like a pointer as far as naive code is concerned, but
* keeps count of how many handles are holding on to the contents
* and deletes it when there are none.
*
* You need to be careful there are no dumb C++ pointers to things which
* are being handled this way.
*
* Things to be handled should implement the same interface as EST_Handleable
* (either by taking that as a superclass or by reimplementing it) and in
* addition define {\tt object_ptr()}. See EST_TBox for an example.
*
* There are two parameter types. In most cases the thing which contains the
* reference count and the data it represents will be the same object, but
* in the case of boxed values it may not be, so you can specify the type
* of both independently.
*
* @see EST_Handleable
* @see EST_TBox
* @see EST_THandle:example
*
* @author Richard Caley <rjc@cstr.ed.ac.uk>
* @version $Id: EST_THandle.h,v 1.5 2006/07/19 21:52:12 awb Exp $
*/
template<class BoxT, class ObjectT>
class EST_THandle {
private:
BoxT *ptr;
public:
EST_THandle(void) { ptr = (BoxT *)NULL; }
EST_THandle(BoxT *p) { if ((ptr=p)) p->inc_refcount(); }
EST_THandle(const EST_THandle &cp) {
ptr=cp.ptr;
if (ptr)
ptr->inc_refcount();
}
~EST_THandle(void) {
if (ptr)
ptr->dec_refcount();
if (ptr && ptr->is_unreferenced())
delete ptr;
}
bool null() const { return ptr == NULL; }
int shareing(void) const { return ptr?(ptr->refcount() > 1):0; }
EST_THandle &operator = (EST_THandle h) {
// doing it in this order means self assignment is safe.
if (h.ptr)
(h.ptr)->inc_refcount();
if (ptr)
ptr->dec_refcount();
if (ptr && ptr->is_unreferenced())
delete ptr;
ptr=h.ptr;
return *this;
}
// If they manage to get hold of one...
// Actually usually used to assign NULL and so (possibly) deallocate
// the object currently pointed to.
EST_THandle &operator = (BoxT *t_ptr) {
// doing it in this order means self assignment is safe.
if (t_ptr)
t_ptr->inc_refcount();
if (ptr)
ptr->dec_refcount();
if (ptr && ptr->is_unreferenced())
delete ptr;
ptr=t_ptr;
return *this;
}
operator ObjectT *() {
return ptr?(ptr->object_ptr()):(ObjectT *)NULL;
}
operator const ObjectT *() const {
return ptr?(ptr->object_ptr()):(const ObjectT *)NULL;
}
int operator == (const BoxT *p) const { return ptr == p; }
int operator != (const BoxT *p) const { return !(*this == p); }
const ObjectT& operator *() const { return *(ptr->object_ptr()); }
ObjectT& operator *() { return *(ptr->object_ptr()); }
const ObjectT* operator ->() const { return (ptr->object_ptr()); }
ObjectT* operator ->() { return (ptr->object_ptr()); }
friend int operator == (const EST_THandle< BoxT, ObjectT > &a, const EST_THandle< BoxT, ObjectT > & b)
{return a.ptr==b.ptr; }
friend int operator != (const EST_THandle< BoxT, ObjectT > &a, const EST_THandle< BoxT, ObjectT > & b)
{ return !( a==b ); }
friend ostream & operator << (ostream &s, const EST_THandle< BoxT, ObjectT > &a)
{ return s << "HANDLE"; }
};
#endif