Hi Everyone,
I was overloading the operator= function as a class member function,
#include <iostream.h>
class A
{
int value;
public : A& operator = (const A& ref)
{
this->value = ref.value;
cout<<"in"<<endl;
return *this;
}
A(int value) : value(value)
{
}
A(const A& copy)
{
this->value = copy.value;
cout<<"copy with a value of "<<copy.value<<endl;
}
void print()
{
cout<<"value is"<<value<<endl;
}
};
int main()
{
A obj(0);
A obj1(10);
A obj2(20);
obj.print();
obj1.print();
obj2.print();
obj1 = obj = obj2;
obj.print();
obj.print();
obj2.print();
}
It works fine and all objects have a value of 20 as expected...
Next i changed the return type of the operator = function, like
so that it returns an object instead of a reference...
A operator = (const A& ref)
{
this->value = ref.value;
cout<<"in"<<endl;
return *this;
}
And as the object is returned by value, each call to operator= also
invokes the copy constructor for the temp object, which is again
passed to the operator= function, and the end result happens to be
same as with operator= function which returns a reference.
Is there any thumb rule while overloading operators, when we have
alternative implementations?
Thanks in advance!!! 7 1702
On Dec 20, 6:43 pm, Rahul <sam_...@yahoo.co.inwrote:
Is there any thumb rule while overloading operators, when we have
alternative implementations?
Yes. Member operators are generally expected to return a reference to
themselves (at least, that's my undestanding). This provides the
expected behaviour when chaining operators. For example:
MyClass a, b, c;
// Presumably do stuff with c here, then.....
a = b = c;
No extra copies are involved when operator=() returns by reference.
--
Computational Modeling, CSIRO (CMIS)
Melbourne, Australia
Why do you want to return by value from operator=()?
On Dec 20, 1:52 pm, Craig Scott <audiofana...@gmail.comwrote:
On Dec 20, 6:43 pm, Rahul <sam_...@yahoo.co.inwrote:
Is there any thumb rule while overloading operators, when we have
alternative implementations?
Yes. Member operators are generally expected to return a reference to
themselves (at least, that's my undestanding). This provides the
expected behaviour when chaining operators. For example:
MyClass a, b, c;
// Presumably do stuff with c here, then.....
a = b = c;
No extra copies are involved when operator=() returns by reference.
--
Computational Modeling, CSIRO (CMIS)
Melbourne, Australia
Why do you want to return by value from operator=()?
But the chaining works even when operator= returns a copy too...
On 2007-12-20 02:43:29 -0500, Rahul <sa*****@yahoo.co.insaid:
Hi Everyone,
I was overloading the operator= function as a class member function,
#include <iostream.h>
class A
{
int value;
public : A& operator = (const A& ref)
{
this->value = ref.value;
cout<<"in"<<endl;
return *this;
}
A(int value) : value(value)
{
}
A(const A& copy)
{
this->value = copy.value;
cout<<"copy with a value of "<<copy.value<<endl;
}
void print()
{
cout<<"value is"<<value<<endl;
}
};
int main()
{
A obj(0);
A obj1(10);
A obj2(20);
obj.print();
obj1.print();
obj2.print();
obj1 = obj = obj2;
obj.print();
obj.print();
obj2.print();
}
It works fine and all objects have a value of 20 as expected...
Next i changed the return type of the operator = function, like
so that it returns an object instead of a reference...
A operator = (const A& ref)
{
this->value = ref.value;
cout<<"in"<<endl;
return *this;
}
And as the object is returned by value, each call to operator= also
invokes the copy constructor for the temp object, which is again
passed to the operator= function, and the end result happens to be
same as with operator= function which returns a reference.
Is there any thumb rule while overloading operators, when we have
alternative implementations?
Thanks in advance!!!
I can't see any legitimate situation where you would prefer to return a
value instead of a reference here.
So, I would say, between those two options, always choose return by
reference since it avoids a copy constructor invocation.
--
-kira
On Dec 20, 10:40 am, Kira Yamato <kira...@earthlink.netwrote:
On 2007-12-20 02:43:29 -0500, Rahul <sam_...@yahoo.co.insaid:
[...]
Is there any thumb rule while overloading operators, when we
have alternative implementations?
Thanks in advance!!!
I can't see any legitimate situation where you would prefer to
return a value instead of a reference here.
So, I would say, between those two options, always choose
return by reference since it avoids a copy constructor
invocation.
The general rule is to be as much like the built-in operators as
possible. This basically means that if the built-in operator is
an lvalue, return a reference, and if it is not, return a value.
Since = (and all of the <op>=) are lvalues, you return a
reference.
Note too that if there are implicit conversions to the data
type, you probably want the operators which don't require an
lvalue to be non-members.
--
James Kanze (GABI Software) email:ja*********@gmail.com
Conseils en informatique orientée objet/
Beratung in objektorientierter Datenverarbeitung
9 place Sémard, 78210 St.-Cyr-l'École, France, +33 (0)1 30 23 00 34
On 2007-12-20 04:49:53 -0500, James Kanze <ja*********@gmail.comsaid:
On Dec 20, 10:40 am, Kira Yamato <kira...@earthlink.netwrote:
>On 2007-12-20 02:43:29 -0500, Rahul <sam_...@yahoo.co.insaid:
[...]
>>Is there any thumb rule while overloading operators, when we have alternative implementations?
>>Thanks in advance!!!
>I can't see any legitimate situation where you would prefer to return a value instead of a reference here.
>So, I would say, between those two options, always choose return by reference since it avoids a copy constructor invocation.
The general rule is to be as much like the built-in operators as
possible. This basically means that if the built-in operator is
an lvalue, return a reference, and if it is not, return a value.
Since = (and all of the <op>=) are lvalues, you return a
reference.
Note too that if there are implicit conversions to the data
type, you probably want the operators which don't require an
lvalue to be non-members.
Can a temporary object be an lvalue? The following code compiles fine
under g++ 4.0.1:
int main()
{
class T {};
T() = T();
return 0;
}
--
-kira
On Dec 20, 11:53 am, Kira Yamato <kira...@earthlink.netwrote:
On 2007-12-20 04:49:53 -0500, James Kanze
<james.ka...@gmail.comsaid:
On Dec 20, 10:40 am, Kira Yamato <kira...@earthlink.netwrote:
On 2007-12-20 02:43:29 -0500, Rahul <sam_...@yahoo.co.insaid:
[...]
>Is there any thumb rule while overloading operators, when we have alternative implementations?
>Thanks in advance!!!
I can't see any legitimate situation where you would prefer to
return a value instead of a reference here.
So, I would say, between those two options, always choose
return by reference since it avoids a copy constructor
invocation.
The general rule is to be as much like the built-in operators as
possible. This basically means that if the built-in operator is
an lvalue, return a reference, and if it is not, return a value.
Since = (and all of the <op>=) are lvalues, you return a
reference.
Note too that if there are implicit conversions to the data
type, you probably want the operators which don't require an
lvalue to be non-members.
Can a temporary object be an lvalue?
Not normally. The standard doesn't actually make an explicit
equivalence, but most of the cases listed in §12.2 where a
temporary comes into existence involve expressions which are
rvalues.
The following code compiles fine under g++ 4.0.1:
int main()
{
class T {};
T() = T();
return 0;
}
Yes. The reason is that there is no problem calling a member
function on a temporary, and for class types, operator= is a
member function.
To return to my initial point: there is no way to restrict a
user defined operator to lvalues. By making it a member,
however, you do prevent conversions: built-in operators
which require lvalues will not accept the results of
conversions, and those that do not require lvalues do.
At least, that's the commonly accepted point of view. On
thinking about it, however: if you define something like
operator+= as a non-member (friend), then the first argument
will be a non-const reference (otherwise you cannot possibly
give it the semantics it should have). And you can't initialize
a non-const reference with a temporary. Which would argue for
making operators requiring an lvalue free functions, and not
members. (Except for operator=, which for other reasons cannot
be a free function.)
Note that the issue is far from moot. Suppose you want an
iterator to the last element in a container you know is not
empty, and that has at least a bidirectional iterator (e.g. any
of the standard sequences). You might be tempted to write:
-- container.end()
. If the operator-- function of the iterator is a member (which
it is in most, if not all, current implementations), this works.
If it is not (or if the "iterator" is just a typedef for a
pointer, which was the case in a lot of early implementations of
std::vector), then it doesn't. (Since the abstraction of STL
iterators is a pointer, we'd really like for it not to work.)
So it's worth thinking about, even if the currently accepted
best practice is as I originally stated: that operators which
require lvalues be members, and operators that don't be free
functions. (And of course, when you deviate from currently
accepted best practices, aka the standard way of doing things,
you raise questions in your readers' minds. It's better not to
without very strong reasons.)
--
James Kanze (GABI Software) email:ja*********@gmail.com
Conseils en informatique orientée objet/
Beratung in objektorientierter Datenverarbeitung
9 place Sémard, 78210 St.-Cyr-l'École, France, +33 (0)1 30 23 00 34
On 2007-12-21 04:27:06 -0500, James Kanze <ja*********@gmail.comsaid:
On Dec 20, 11:53 am, Kira Yamato <kira...@earthlink.netwrote:
>On 2007-12-20 04:49:53 -0500, James Kanze <james.ka...@gmail.comsaid:
>>On Dec 20, 10:40 am, Kira Yamato <kira...@earthlink.netwrote: On 2007-12-20 02:43:29 -0500, Rahul <sam_...@yahoo.co.insaid:
>>[...] Is there any thumb rule while overloading operators, when we have alternative implementations?
>>>>Thanks in advance!!!
>>>I can't see any legitimate situation where you would prefer to return a value instead of a reference here.
>>>So, I would say, between those two options, always choose return by reference since it avoids a copy constructor invocation.
>>The general rule is to be as much like the built-in operators as possible. This basically means that if the built-in operator is an lvalue, return a reference, and if it is not, return a value. Since = (and all of the <op>=) are lvalues, you return a reference.
>>Note too that if there are implicit conversions to the data type, you probably want the operators which don't require an lvalue to be non-members.
>Can a temporary object be an lvalue?
Not normally. The standard doesn't actually make an explicit
equivalence, but most of the cases listed in §12.2 where a
temporary comes into existence involve expressions which are
rvalues.
>The following code compiles fine under g++ 4.0.1:
>int main() { class T {}; T() = T(); return 0; }
Yes. The reason is that there is no problem calling a member
function on a temporary, and for class types, operator= is a
member function.
To return to my initial point: there is no way to restrict a
user defined operator to lvalues. By making it a member,
however, you do prevent conversions: built-in operators
which require lvalues will not accept the results of
conversions, and those that do not require lvalues do.
At least, that's the commonly accepted point of view. On
thinking about it, however: if you define something like
operator+= as a non-member (friend), then the first argument
will be a non-const reference (otherwise you cannot possibly
give it the semantics it should have). And you can't initialize
a non-const reference with a temporary. Which would argue for
making operators requiring an lvalue free functions, and not
members. (Except for operator=, which for other reasons cannot
be a free function.)
Note that the issue is far from moot. Suppose you want an
iterator to the last element in a container you know is not
empty, and that has at least a bidirectional iterator (e.g. any
of the standard sequences). You might be tempted to write:
-- container.end()
. If the operator-- function of the iterator is a member (which
it is in most, if not all, current implementations), this works.
If it is not (or if the "iterator" is just a typedef for a
pointer, which was the case in a lot of early implementations of
std::vector), then it doesn't. (Since the abstraction of STL
iterators is a pointer, we'd really like for it not to work.)
So it's worth thinking about, even if the currently accepted
best practice is as I originally stated: that operators which
require lvalues be members, and operators that don't be free
functions. (And of course, when you deviate from currently
accepted best practices, aka the standard way of doing things,
you raise questions in your readers' minds. It's better not to
without very strong reasons.)
Very interesting read.
So, this is telling me that we should do things in your suggested way:
that is, declare operators that should require lvalues be free
functions and declare its first parameter a non-const reference.
Well, even if it is not standard practice, I'll start doing that.
--
-kira This thread has been closed and replies have been disabled. Please start a new discussion. Similar topics
by: Edward Diener |
last post by:
Is there a way to override the default processing of the assignment operator
for one's own __value types ? I realize I can program my own Assign method,
and provide that for end-users of my class,...
|
by: Pmb |
last post by:
I've been working on creating a Complex class for my own learning purpose
(learn through doing etc.). I'm once again puzzled about something. I can't
figure out how to overload the assignment...
|
by: gorda |
last post by:
Hello,
I am playing around with operator overloading and inheritence,
specifically overloading the + operator in the base class and its
derived class.
The structure is simple: the base class...
|
by: pmatos |
last post by:
Hi all,
I'm overloading operator<< for a lot of classes. The question is about
style. I define in each class header the prototype of the overloading
as a friend. Now, where should I define the...
|
by: carlos |
last post by:
Curious:
Why wasnt a primitive exponentiation operator not added to C99?
And, are there requests to do so in the next std revision?
Justification for doing so:
C and C++ are increasingly used...
|
by: karthik |
last post by:
The * operator behaves in 2 different ways. It is used as the value at
address operator as well as the multiplication operator. Does this mean
* is overloaded in c?
|
by: Jerry Fleming |
last post by:
As I am newbie to C++, I am confused by the overloading issues. Everyone
says that the four operators can only be overloaded with class member
functions instead of global (friend) functions: (), ,...
|
by: y-man |
last post by:
Hi,
I am trying to get an overloaded operator to work inside the class it
works on. The situation is something like this:
main.cc:
#include "object.hh"
#include "somefile.hh"
object obj,...
|
by: sturlamolden |
last post by:
Python allows the binding behaviour to be defined for descriptors,
using the __set__ and __get__ methods. I think it would be a major
advantage if this could be generalized to any object, by...
|
by: Wayne Shu |
last post by:
Hi everyone, I am reading B.S. 's TC++PL (special edition).
When I read chapter 11 Operator Overloading, I have two questions.
1. In subsection 11.2.2 paragraph 1, B.S. wrote "In particular,...
|
by: lllomh |
last post by:
Define the method first
this.state = {
buttonBackgroundColor: 'green',
isBlinking: false, // A new status is added to identify whether the button is blinking or not
}
autoStart=()=>{
|
by: DJRhino |
last post by:
Was curious if anyone else was having this same issue or not....
I was just Up/Down graded to windows 11 and now my access combo boxes are not acting right. With win 10 I could start typing...
|
by: tracyyun |
last post by:
Hello everyone,
I have a question and would like some advice on network connectivity. I have one computer connected to my router via WiFi, but I have two other computers that I want to be able to...
|
by: giovanniandrean |
last post by:
The energy model is structured as follows and uses excel sheets to give input data:
1-Utility.py contains all the functions needed to calculate the variables and other minor things (mentions...
|
by: Teri B |
last post by:
Hi, I have created a sub-form Roles. In my course form the user selects the roles assigned to the course.
0ne-to-many. One course many roles.
Then I created a report based on the Course form and...
|
by: nia12 |
last post by:
Hi there,
I am very new to Access so apologies if any of this is obvious/not clear.
I am creating a data collection tool for health care employees to complete. It consists of a number of...
|
by: NeoPa |
last post by:
Introduction
For this article I'll be focusing on the Report (clsReport) class. This simply handles making the calling Form invisible until all of the Reports opened by it have been closed, when it...
|
by: isladogs |
last post by:
The next online meeting of the Access Europe User Group will be on Wednesday 6 Dec 2023 starting at 18:00 UK time (6PM UTC) and finishing at about 19:15 (7.15PM).
In this month's session, Mike...
|
by: GKJR |
last post by:
Does anyone have a recommendation to build a standalone application to replace an Access database? I have my bookkeeping software I developed in Access that I would like to make available to other...
| |