haiku/headers/os/support/Referenceable.h
Adrien Destugues 49e8a3c652 BReferencable: implement const references with specialization
This is simpler and cleaner than my previous attempt. Thanks to Ingo for
suggesting this.
2015-01-27 10:31:08 +01:00

268 lines
3.9 KiB
C++

/*
* Copyright 2004-2010, Ingo Weinhold, ingo_weinhold@gmx.de.
* Distributed under the terms of the MIT License.
*/
#ifndef _REFERENCEABLE_H
#define _REFERENCEABLE_H
#include <SupportDefs.h>
// #pragma mark - BReferenceable
class BReferenceable {
public:
BReferenceable();
virtual ~BReferenceable();
// acquire and release return
// the previous ref count
int32 AcquireReference();
int32 ReleaseReference();
int32 CountReferences() const
{ return fReferenceCount; }
protected:
virtual void FirstReferenceAcquired();
virtual void LastReferenceReleased();
protected:
int32 fReferenceCount;
};
// #pragma mark - BReference
template<typename Type = BReferenceable>
class BReference {
public:
BReference()
:
fObject(NULL)
{
}
BReference(Type* object, bool alreadyHasReference = false)
:
fObject(NULL)
{
SetTo(object, alreadyHasReference);
}
BReference(const BReference<Type>& other)
:
fObject(NULL)
{
SetTo(other.Get());
}
template<typename OtherType>
BReference(const BReference<OtherType>& other)
:
fObject(NULL)
{
SetTo(other.Get());
}
~BReference()
{
Unset();
}
void SetTo(Type* object, bool alreadyHasReference = false)
{
if (object != NULL && !alreadyHasReference)
object->AcquireReference();
Unset();
fObject = object;
}
void Unset()
{
if (fObject) {
fObject->ReleaseReference();
fObject = NULL;
}
}
Type* Get() const
{
return fObject;
}
Type* Detach()
{
Type* object = fObject;
fObject = NULL;
return object;
}
Type& operator*() const
{
return *fObject;
}
Type* operator->() const
{
return fObject;
}
operator Type*() const
{
return fObject;
}
BReference& operator=(const BReference<Type>& other)
{
SetTo(other.fObject);
return *this;
}
BReference& operator=(Type* other)
{
SetTo(other);
return *this;
}
template<typename OtherType>
BReference& operator=(const BReference<OtherType>& other)
{
SetTo(other.Get());
return *this;
}
bool operator==(const BReference<Type>& other) const
{
return fObject == other.fObject;
}
bool operator==(const Type* other) const
{
return fObject == other;
}
bool operator!=(const BReference<Type>& other) const
{
return fObject != other.fObject;
}
bool operator!=(const Type* other) const
{
return fObject != other;
}
private:
Type* fObject;
};
// #pragma mark - BReference<const>
template<typename Type>
class BReference<const Type> {
public:
BReference(Type* object, bool alreadyHasReference = false)
:
fReference(object, alreadyHasReference)
{
}
BReference(const BReference<const Type>& other)
:
fReference(other)
{
}
template<typename OtherType>
BReference(const BReference<OtherType>& other)
:
fReference(other.Get())
{
}
void SetTo(Type* object, bool alreadyHasReference = false)
{
fReference.SetTo(object, alreadyHasReference);
}
void Unset()
{
fReference.Unset();
}
const Type* Get() const
{
return fReference.Get();
}
const Type* Detach()
{
return fReference.Detach();
}
const Type& operator*() const
{
return *fReference;
}
const Type* operator->() const
{
return fReference.Get();
}
operator const Type*() const
{
return fReference.Get();
}
BReference& operator=(const BReference<const Type>& other)
{
fReference = other.fReference;
}
BReference& operator=(Type* other)
{
fReference = other;
}
template<typename OtherType>
BReference& operator=(const BReference<OtherType>& other)
{
fReference = other.Get();
}
bool operator==(const BReference<const Type>& other) const
{
return fReference == other.Get();
}
bool operator==(const Type* other) const
{
return fReference == other;
}
bool operator!=(const BReference<const Type>& other) const
{
return fReference != other.Get();
}
bool operator!=(const Type* other) const
{
return fReference != other;
}
private:
BReference<Type> fReference;
};
#endif // _REFERENCEABLE_H