Re: C++ Frequently Questioned Answers
On Nov 6, 6:33 am, Yossi Kreinin <yossi.krei...@gmail.com> wrote:
On Nov 5, 5:08 pm, Kirit S?lensminde <kirit.saelensmi...@gmail.com>
wrote:
Yossi Kreinin wrote:
Here's real code from a social network back-end :)
class Lamer { public: const LameComments& getLameComments(); };
void getTheMostLameComments(
vector<const LameComments*>& comments, //which type would you use?
const vector<Lamer>& lamers
)
{
//or we could use iterators...
for(int i=0; i<(int)lamers.size(); ++i)
{
const LameComments& lc = lamers[i].getLameComments();
if(lc.areTotallyPointless()) {
//lamers write lots of comments, better copy
//by reference... Has to be const - can't modify
//the lamer's precious comments, and we can't
//have vectors of references, so it's either a dumb
//const pointer or some non-standard smart pointer
comments.push_back(&lc);
}
}
}
I think this is a pretty common pattern with code massaging data
structures with even minor levels of nesting/indirection.
As you've discovered, just because it is a common pattern doesn't make
it good.
The thing that I want to compute is certainly not "bad" by itself - it
is a completely trivial filter, it should be trivial to implement, and
it is trivial to implement in all popular programming languages. It is
also pretty trivial in C++, except for the const part; my point was to
show that sometimes you naturally end up with vectors of const
pointers in very simple code, contrary to the arguments in this NG.
You've completely lost me in that case. Of course the pointers should
point to a constant object - that's exactly what your interface says
the objects are (here's your code):
class Lamer { public: const LameComments& getLameComments(); };
If you want to store pointers to mutable LameComments then this
interface should be fixed. As it is, to allow you to get a
LameComments* from a const LameComments& without a cast really would
be broken.
This works:
LameComments foo, bar;
std::vector< const LameComments * > comments( 1 );
comments[ 0 ] = &foo;
comments[ 0 ] = &bar;
It is the LameComment /pointed to/ that is const, /not/ the pointer
and this is a direct consequence of the interface you have defined. I
can't see any problem with C++ here.
The only other odd const behaviour is that due to you placing the
Lamer objects /inside/ the vector (which is const), rather than having
a pointer to them in the vector the comment accessor needs to be
const. Is this the problem you have? If so I address it in more detail
below.
I think we should all ignore the completely unnecessary C cast and
move on.
I don't know which cast you mean; the code has no cast.
If you look closely i is declared with the wrong type. It should be
std::size_t not int. This leads to the C cast in the for condition.
This is a filter on top of a data structure. The C++ way of doing this
is to write an iterator that understands the filter rule. A bit more
advanced is to write an iterator adaptor which can be used to wrap
both const and non-const iterators on the underlying data structure.
I'm sure there's an implementation that takes a plug-in filter
predicate somewhere already.
They're not hard to write and have O(1) memory overhead compared to
the code shown above. They also side step many of the problems you
describe with the data types, const handling etc.
As you mention yourself later, sometimes it's a bad idea since I
really want to create a container in memory. For example, there could
be lots of Lamer objects but only few comments which are
isTotallyPointless, and these comments later undergo heavy processing.
I wouldn't want to scan through the whole data structure again and
again.
Still, while we're at the process of exchanging "real code", it would
be nice if you contributed an example implementation of your idea for
the cases when it is appropriate in terms of run time (there are few
Lamer objects and/or most comments are isTotallyPointless). The
readers could then judge whether what you call "The C++ way of doing
this" (trivial) thing is really "not hard to write" compared to the
(primitive as I think it should be) code I posted. BTW, here's how you
do it in Python - an example of a language with built-in support for
this kind of thing:
comments = [l.getLameComments() for l in lamers \
if l.getLameComments().isTotallyPointless()]
Here is a O(1) iterator based on your original example:
class lame_iterator {
std::vector<Lamer>::const_iterator m_pos;
const std::vector<Lamer>::const_iterator m_end;
void find_lame() {
while ( m_pos != m_end && !m_pos-
getLameComments().areTotallyPointless() )
++m_pos;
}
public:
lame_iterator( const std::vector< Lamer > &lamers )
: m_pos( lamers.begin() ), m_end( lamers.end() ) {
find_lame();
}
const LameComments& operator *() const {
return m_pos->getLameComments();
}
void operator ++() {
++m_pos;
find_lame();
}
bool operator !=( const lame_iterator &end ) {
return m_end != end.m_end;
}
};
It's not as simple as the Python case, but it has O(1) memory overhead
(win some, lose some - this is not new in any language). Your
implementation is about as simple as the Python already and no less
problematic.
The above isn't a full iterator implementation. If you were willing to
change the design slightly (debug it?) then I think the Boost.Iterator
library contains the general iterator building blocks needed (i.e.
boost::filter_iterator).
If you really need a copy of part of the container then
std::remove_copy_if is what you want. Again it sidesteps all of the
type problems, but I'd have to say that std::copy_if would be a more
useful name to have.
But I /can't/ use remove_copy_if or otherwise modify the container,
because it's typed const vector<Lamer>&, bringing us back to the
subject of the downsides of C++ const correctness. To use
remove_copy_if, I'd have to create a vector of const pointers to Lamer
objects (or LameComments objects or I could use
vector<Lamer>::const_iterator instead of const Lamer*), since this is
the only kind of pointer to the objects I can obtain without a cast.
And having a vector of const pointers or const_iterators brings us
back to square one. Or I'd have to copy the Lamer objects or all of
the LameComments objects, which has its run time cost.
Well of course the things inside a const vector are also const. Are
you trying to argue that the values inside a const vector should be
mutable? I think what you want is this: const vector<Lamer*> &. Now
the pointers inside the vector are const, but the objects they point
to are not. The indirection means that the vector is const, the
pointers inside it are const, but the Lamer objects that are
referenced are /not/ const.
What I'm saying is, to work around this you need to const_cast
pointers or copy objects or copy code (ultimately have two functions
somewhere later, one for vector<const T*> and another one for
vector<T*>). All of these mean that const gets in your way in this
example.
I'm not sure that you do if you adjust your expectation and see what
const does rather than what you wish it did.
K
--
[ See http://www.gotw.ca/resources/clcm.htm for info about ]
[ comp.lang.c++.moderated. First time posters: Do this! ]