A set of APIs that I commonly use follow a linked-list pattern:
struct SomeObject
{
const char* some_value;
const char* some_other_value;
SomeObject* next;
}
LONG GetObjectList( SomeObject** list );
void FreeObjectList( SomeObject* list );
This API is not mine and I cannot change it.
So, I’d like to encapsulate their construction/destruction, access, and add iterator support. My plan is to do something like this:
/// encapsulate access to the SomeObject* type
class MyObject
{
public:
MyObject() : object_( NULL ) { };
MyObject( const SomeObject* object ) : object_( object ) { };
const char* SomeValue() const
{
return NULL != object_ ? object_->some_value : NULL;
};
const char* SomeValue() const
{
return NULL != object_ ? object_->some_other_value : NULL;
};
private:
SomeObject* object_;
}; // class MyObject
bool operator==( const MyObject& i, const MyObject& j )
{
return // some comparison algorithm.
};
/// provide iterator support to SomeObject*
class MyObjectIterator
: public boost::iterator_adaptor< MyObjectIterator,
MyObject*,
boost::use_default,
boost::forward_traversal_tag >
{
public:
// MyObjectIterator() constructors
private:
friend class boost::iterator_core_access;
// How do I cleanly give the iterator access to the underlying SomeObject*
// to access the `next` pointer without exposing that implementation detail
// in `MyObject`?
void increment() { ??? };
};
/// encapsulate the SomeObject* creation/destruction
class MyObjectList
{
public:
typedef MyObjectIterator const_iterator;
MyObjectList() : my_list_( MyObjectList::Create(), &::FreeObjectList )
{
};
const_iterator begin() const
{
// How do I convert a `SomeObject` pointer to a `MyObject` reference?
return MyObjectIterator( ??? );
};
const_iterator end() const
{
return MyObjectIterator();
};
private:
static SomeObject* Create()
{
SomeObject* list = NULL;
GetObjectList( &list );
return list;
};
boost::shared_ptr< void > my_list_;
}; // class MyObjectList
My two questions are:
-
How do I cleanly give
MyObjectIteratoraccess to the underlyingSomeObjectto access thenextpointer in the linked list without exposing that implementation detail inMyObject? -
In
MyObjectList::begin(), how do I convert aSomeObjectpointer to aMyObjectreference?
Thanks,
PaulH
Edit: The linked-list APIs I’m wrapping are not mine. I cannot change them.
You would make
MyObjectIteratora friend ofMyObject. I don’t see any better way. And really I think it’s reasonable that iterators get whatever special friend access is necessary for them to perform their duties.You don’t seem to have considered how and where your
MyObjectinstance are going to be stored. Or perhaps that’s what this question is coming out of. It seems like you would have to have a separate linked list of MyObjects in your MyObjectList. Then at leastMyObjectList::begin()can just grab the firstMyObjectof your internal linked list of them. And if the only operations that may modify or rearrange this list only ever happen through the iterators you provide, then you can problem keep these lists in synch without too much trouble. Otherwise, if there are functions in the API you’re using that take the rawSomeObjectlinked list and manipulate it, then you may have trouble.I can see why you’ve tried to design this scheme, but having separate MyObjects that point to SomeObjects even through SomeObjects themselves make up the real list structure…. That is not an easy way to wrap a list.
The simplest alternative is just to do away with MyObject completely. Let your iterators work against SomeObject instances directly and return those when dereferenced. Sure that does expose SomeObject to the outside, particularly its
nextmember. But is that really a big enough problem to justify a more complex scheme to wrap it all up?Another way to deal with might be to have
MyObjectprivately inherit fromSomeObject. Then each SomeObject instance can be downcast as a reference to a MyObject instance and given to the outside world that way, thus hiding the implemtation details of SomeObject and only exposing the desired public member functions. The standard probably doesn’t guarantee this will work, but in practice since they’ll likely have the exact same memory layouts you may be able to get away with it. However, I’m not sure that I would actually ever try such a thing in a real program unless absolutely necessary.The last alternative I can think of is just to transfer the data into a list of more convenient data structures after being given to you by this API. And then of course transfer it back into a raw SomeObject list only if necessary to pass it back to the API. Just make your own
SomeObjectDataor whatever to store the strings and put them in astd::list. Whether or not this is actually feasible for you really depends on how this data is used in the context of the API you’ve mentioned. If there are other API functions that modify SomeObject lists and you need to use them frequently, then constantly converting SomeObject lists to and fromstd::list<SomeObjectData>could be annoying.