Re: Temporary objects and operators overloading
On Jul 6, 8:51 pm, SG <s.gesem...@gmail.com> wrote:
<snip>
Seems like G++ is not a halfway decent compiler, then. The issue here
is that you have an expression in the return statement -- a function
call returning a reference -- that is not guaranteed to refer to the
object called "result" just by looking at the function's signature.
So, unless operator+= is inlined and the compiler is smart enough to
figure out that the reference refers to "result", it cannot apply the
return value optimization.
I actually tested it a couple of weeks ago with G++ 4.3.3:
Foo operator+(Foo const & lhs, Foo const & rhs){
return Foo(lhs) += rhs; // No RVO is done
}
Foo operator+(Foo const & lhs, Foo const & rhs){
Foo result(lhs);
return result += rhs; // No NRVO is done
}
Foo operator+(Foo const & lhs, Foo const & rhs){
Foo result(lhs);
result += rhs;
return result; // NRVO is done
}
Even with an inline operator+= doesn't help here.
Cheers!
SG
Let's take a concrete example with a concrete compiler:
g++ 4.1.2, i386-redhat-linux
compiled with following options:
-g -O2 -Wall -Wno-long-long --pedantic test.cpp
the class:
class A
{
public:
A(int in) : n_(in)
{
cout << "Constructor: input = " << in << endl;
}
A(A const & in) : n_(in.n_)
{
cout << "Copy constructor: input = "
<< in.n_ << endl;
}
A & operator+= (A const & in)
{
n_ += in.n_;
return *this;
}
operator int() const { return n_; }
private:
int n_;
};
operator= versions:
(1)
A operator+ (A left, A right)
{
left += right;
return left;
}
(2)
A operator+ (A left, A const & right)
{
left += right;
return left;
}
(3)
A operator+ (A const & left, A const & right)
{
return A((int)left + (int)right);
}
Test Group 1
Running the following code:
A a = A(1) + A(2) + A(4) + A(5);
Test 1.1 with operator + defined as (1):
Ouptut:
Constructor: input = 5
Constructor: input = 4
Constructor: input = 2
Constructor: input = 1
Copy constructor: input = 3
Copy constructor: input = 7
Copy constructor: input = 12
Test 1.2 with operator= defined as (2):
Output:
Constructor: input = 5
Constructor: input = 4
Constructor: input = 2
Constructor: input = 1
Copy constructor: input = 3
Copy constructor: input = 7
Copy constructor: input = 12
Test 1.3 with operator+ defined as (3):
Output:
Constructor: input = 5
Constructor: input = 4
Constructor: input = 2
Constructor: input = 1
Constructor: input = 3
Constructor: input = 7
Constructor: input = 12
Test Group 2
Running the following code:
A a1(1);
A a2(2);
A a4(4);
A a5(5);
A a = a1 + a2 + a4 + a5;
Test 2.1 with operator= defined as
A operator+ (A left, A right)
{
left += right;
return left;
}
Output:
Constructor: input = 1
Constructor: input = 2
Constructor: input = 4
Constructor: input = 5
Copy constructor: input = 5
Copy constructor: input = 4
Copy constructor: input = 2
Copy constructor: input = 1
Copy constructor: input = 3
Copy constructor: input = 7
Copy constructor: input = 12
Test 2.2 with operator= defined as
A operator+ (A left, A const & right)
{
left += right;
return left;
}
Output:
Constructor: input = 1
Constructor: input = 2
Constructor: input = 4
Constructor: input = 5
Copy constructor: input = 1
Copy constructor: input = 3
Copy constructor: input = 7
Copy constructor: input = 12
Test 3.3 with operator= defined as
A operator+ (A const & left, A const & right)
{
return A((int)left + (int)right);
}
Output:
Constructor: input = 1
Constructor: input = 2
Constructor: input = 4
Constructor: input = 5
Constructor: input = 3
Constructor: input = 7
Constructor: input = 12
So it looks like this compiler is perfectly capable
of doing both RVO and copy elision when only temporaries
are use. All three version of operator= produced
identical results when called only with temporaries.
Obviously it can't do copy elision when
operator+ is called with named variables. This is
when versions that take const ref really shine -
operator= (val, val) = 7 constructors
operator= (val, &val) = 4 constructors
operator= (&val, &val) = 3 constructors
HTH,
Andy.
--
[ See http://www.gotw.ca/resources/clcm.htm for info about ]
[ comp.lang.c++.moderated. First time posters: Do this! ]