Re: The dangers of returning const references to values held in a map
On Tue, 23 Oct 2012 10:41:45 -0700 (PDT)
James Lothian <james@firstnamelastname.freeserve.co.uk> wrote:
James K. Lowden wrote:
[snip]
For example, it's impossible to say definitively if map::insert or
map::operator[] is more efficient. You would think, given their
different semantics, that operator[] would always be preferred for
reasons of clarity unless it was important for insert to fail.
But every time I've asked someone why he chose map::insert,
efficiency was the answer.
Hmm.
class MyThing { /* content elided */ };
std::map<std::string, MyThing*> myMap;
myMap["HelloMum"] = new MyThing;
What gets left in the map if MyThing's constructor throws?
map::operator[] calls map::insert for a default-contructed MyThing,
and returns a reference to the mapped_type object. If that
construction fails, the map is unmodified.
The assignment calls MyThing::operator=. If new throws an exeption,
the assignment never happens.
In your particular example, both constructors are the default
constructor. It's possible one would succeed and the other fail but
in this particular case the effect on the map is (logically) the same!
I think your example is more interesting with
myMap["hello"] = new MyThing( many, interesting, parameters );
because then a default MyThing was added as the referent. However, the
problem is simply solved at very little cost:
MyThing *p = new MyThing( many, interesting, parameters );
myMap["hello"] = p;
which I assert is much easier to read than
myMap.insert(
make_pair(
"hello",
new MyThing( many, interesting, parameters )
)
);
--jkl
--
[ See http://www.gotw.ca/resources/clcm.htm for info about ]
[ comp.lang.c++.moderated. First time posters: Do this! ]