ostream_iterator and a collection of pointers

S

ShaunJ

I would like to use an ostream_iterator to print the contents of a
collection of pointers. I don't want to print the pointers, however,
but the objects to which they point.

Thanks,
Shaun
 
J

Jerry Coffin

I would like to use an ostream_iterator to print the contents of a
collection of pointers. I don't want to print the pointers, however,
but the objects to which they point.

Exactly as stated, I don't think it's possible. If you're willing to
use a proxy class for the pointers, it becomes pretty trivial:

// warning: code only minimally tested.
#include <iostream>

template <class T>
class Proxy {
T ptr;
public:
Proxy(T p) : ptr(p) {}
operator T() { return ptr; }
};

// For this to work 'T' must be a pointer(like) object -- one
// for which unary '*' is supported.
template <class T>
std::eek:stream &operator<<(std::eek:stream &os, Proxy<T> p) {
return os << *p;
}

and a quick demo to show it can do the job:

#ifdef TEST
#include <vector>

int main() {
std::vector<Proxy<int *> > ints;

int a = 0, b=1, c=2;

ints.push_back(&a);
ints.push_back(&b);
ints.push_back(&c);

std::copy(ints.begin(),
ints.end(),
std::eek:stream_iterator<Proxy<int *> >(std::cout, "\n"));
return 0;
}

#endif

I'll leave it to you to decide whether this is really a good idea or
not. It does what you've asked with essentially no overhead, but I'm
still not sure I'd use it in production code. At the very least, I'd
probably rename the Proxy class to something indicating that it's
intended as a proxy for pointers. With support for concepts, we could
include (in real code) some assurance that T is dereferencable, but
as it stands right now, we only have comments to assure that (along
with the fact that otherwise code using it won't compile, but the
error messages may be a bit ugly.
 
J

Jerry Coffin

[ ... ]
// For this to work 'T' must be a pointer(like) object -- one
// for which unary '*' is supported.

Thinking about it a moment longer, in this case we can enforce that
restriction (without Concepts). We just change operator<< to look
like:
template <class T>
std::eek:stream &operator<<(std::eek:stream &os, Proxy<T *> p) {
return os << *p;
}
 
B

Bart van Ingen Schenau

ShaunJ said:
I would like to use an ostream_iterator to print the contents of a
collection of pointers. I don't want to print the pointers, however,
but the objects to which they point.

You could use std::transform to copy the collection to the stream and
dereference the contents in one go:

struct Dereference
{
template <typename T>
const T& operator()(const T* item) const
{ return *item; }
};

std::transform(coll.begin(), coll.end(),
std::ostream_iterator said:
Thanks,
Shaun

Bart v Ingen Schenau
 
S

ShaunJ

You could use std::transform to copy the collection to the stream and
dereference the contents in one go:

struct Dereference
{
template <typename T>
const T& operator()(const T* item) const
{ return *item; }

};

std::transform(coll.begin(), coll.end(),
std::eek:stream_iterator<T>(stream, ""), Dereference());

Fantastic! This is exactly what I was looking for. Thanks,
Shaun
 

Ask a Question

Want to reply to this thread or ask your own question?

You'll need to choose a username for the site, which only take a couple of moments. After that, you can post your question and our members will help you out.

Ask a Question

Members online

No members online now.

Forum statistics

Threads
474,432
Messages
2,571,682
Members
48,796
Latest member
Greg L.

Latest Threads

Top