Re: Why extracting string from stringstream(a) fails?
On Oct 26, 11:02 pm, Maxim Yegorushkin <maxim.yegorush...@gmail.com>
wrote:
On 26/10/09 17:47, James Kanze wrote:
On Oct 26, 4:07 pm, Maxim Yegorushkin<maxim.yegorush...@gmail.com>
wrote:
[...]
The argument passing is defined as copy initialization. And
in C++98 the implementation is allowed to make any number of
copies of an rvalue actual argument passed to 'T const&'
formal argument, or for any copy initialization. Which means
that the type must provide a suitable copy constructor. For
example, that means that you can't do this thing with a
std::auto_ptr. Or a stream.
Even more interesting.
Given the following declaration:
void foo(int& ref);
Could you explain how ref argument can possibly be
copy-initialized please?
There's more to it that Alf revealed. (It wouldn't be the
C++ standard if it were that simple.) It's copy
initialization, so the rules for copy initialization apply.
The rule that requires a copy constructor when copy
initialization is used only applies when initializing a
reference with an rvalue (and this is the only time copies
are allowed). If the initializer is an lvalue, everything
is fine---otherwise, things like:
int i;
int& ri = i;
would have somewhat unexpected semantics.
I've been looking at 8.5 Initializers now. My understanding is
that because that form of initialization which is used for
function arguments is called copy initialization, it implies
that a copy constructor is required regardless of whether an
argument is a reference that can be bound directly.
What part of 8.5 exactly? In =A78.5/12,13, it says that the
initializer must be copied, but the initializer here would be
the T&, the reference. (That does have "interesting"
implications concerning the lifetime of temporaries, but I'm
pretty sure that that isn't intentional.) =A78.5.3/5 is quite
clear: "[...]-- If the initializer expression [...]-- is an
lvalue[...] then the reference is bound directly to the
initializer expression."
It also says that copy elision is a permissible optimization,
but does not require it.
Yes, but since we're talking here about copying the reference, I
don't think that it's too relevant. References are copiable,
regardless of what they refer to. The possible problem
regarding copy is the lifetime of the temporary. Given
something like:
int const& ri = 3;
, the temporary containing the 3 has its lifetime extended to
that of the reference it initializes. If the reference it
initializes is a temporary, which is then copied to initialize
ri, this doesn't help us much.
As I said, I don't think this was intended, and I none of the
compilers I know implement it this way.
My intuitive expectation is that implied by the spirit of C++:
you don't pay for what you don't use. Applied to the
initialization of reference function arguments I would expect
it to require the copy constructor only when it is actually
used. Oh, well ;)
You don't want the legality of a program changed by whether the
compiler does some optional optimization or not.
Alf and Johannes report that initialization of references does
not require a copy constructor in the C++0x standard. I am
glad that they elaborated this case :)
I've still got to examine the new text. The reason for the rule
(I think) is for things like:
struct S { A a; B b; }; // A and B class types...
S f();
B const& rb = f().b;
What should be the lifetime of the S returned by f(). I'd argue
that the current text of the standard says that its lifetime
must end at the end of the full expression. But the only way to
end it, while extending the lifetime of the B temporary bound to
rb, is to copy it out of the S object. (A quick read of the
text in N2914---a recent, but perhaps not the latest,
draft---seems a bit ambiguous. None of the enumerated points
seems to cover this case.)
--
James Kanze