In the Google C++ Style Guide, the section on Operator Overloading recommends against overloading any operators ("except in rare, special circumstances"). Specifically, it recommends:
In particular, do not overload
operator==
oroperator<
just so that your class can be used as a key in an STL container; instead, you should create equality and comparison functor types when declaring the container.
I'm a little fuzzy on what such a functor would look like, but my main question is, why would you want to write your own functors for this? Wouldn't defining operator<
, and using the standard std::less<T>
function, be simpler? Is there any advantage to using one over the other?
Well, according to the webpage you cite, there's not much advantage for functors ("[operators] can fool our intuition into thinking that expensive operations are cheap, built-in operations.")
My felling is that one should strive to make you classes first-class objects as much as possible, which, to me, means having them understand as many operators as sensible.
It's been a while since I written a functor, but it would look something like this:
}
Ironically, a functor also requires overriding an operator (the function call operator -
operator ()
), so I'm not sure what their point is.I think the message behind not defining operator< is that ordering is a property of the collection, not of the object. Different collections of the same objects may have different orderings. So you should use a separate functor used when specifying the type of the collection rather than operator<.
In practice though, a lot of your classes may have a natural ordering and that is the only ordering used in collections in your application. In other cases, the ordering may not even be relevant to the application, just the collection so it can find items later. In these cases, it makes perfect sense to define operator<.
Remember, when we're desiging object models, we're only modelling a subset of the real world. In the real world there may be umpteen different ways to rank objects of the same class, but in the application domain in which we are working there may be one that is relevant.
If code evolves to need a second ordering that is just as relevant as the first, the class should be refactored to remove operator< and to place both ranking functions in separate functors. This shows the intent that no one ranking is more important than the others.
With regard to arithmetic operators, you should not overload these unless you are implementing an arithmetic type.
Of course, there are exceptions to every rule. If you don't know whether or not you should be making an exception, you probably should not. Experience will be your guide.
A functor is a class with an
operator ()
. In this case the method would take two parameters of the type being compared and return abool
result if the first is less than the second.Edit: to build on what James Curran said, you can define your functor inside the class. So, for example:
I probably wouldn't go as far as the Google style guide.
I think that what they are getting at is that when you overload
operator<
andoperator==
, you are making a decision for every use of the type, while the functor types only apply to that collection.If the only thing you need the comparators for is to put the item in a collection, then it's better to have a function specifically for that context rather than the operators that would apply in all contexts.
You may want to sort purchase orders chronologically, but in general, it would make sense to compare them by their total price. If we overload
operator<
to compare dates so that we can load them into a collection, we are introducing the risk that another client may misuse ouroperator<
which they may think compares the total prices.Generally, defining
operator<
is better and simpler.The case where you would want to use functors is when you need multiple ways of comparing a particular type. For example:
In this case there may not be a good "default" way to compare and order people so not defining
operator<
and using functors may be better. You could also say that generally people are ordered by height, and sooperator<
just callsCompareByHeight
, and anyone who needs Person's to be ordered by weight has to useCompareByWeight
explicitly.Often times the problem is that defining the functors is left up to the user of the class, so that you tend to get many redefinitions of the same thing, whenever the class needs to be used in an ordered container.