小括号()操作符函数:
class SpreadsheetCompare{
public:
bool operator()(const QStringList& row1,
const QStringList& row2) const;
enum{ KeyCount = 3 };
int keys[KeyCount];
bool ascending[KeyCount];
};
SpreadsheetCompare类有些特殊,因为它实现了“()”操作符。这样就允许把这个类像函数一样使用。把这样的类称为函数对象(function object),或者称为仿函数(functor)。为了理解仿函数是如何工作的,首先从一个简单的例子开始:
class Square{
public:
int operator()(int x){
return x*x;
}
};
Square类提供了一个函数,operator()(int)函数,它返回其参数的平方值。通过把这个函数命名为operator()(int),而不是将其命名为compute(int)之类的函数,就可以把一个类型为Square的对象当作一个函数。
Square square;
int y = square(5);
// y equals 25
现在,让我们来看一个包括SpreadsheetCompare的实例:
QStringList row1,row2;
SpreadsheetCompare compare;
…
if(compare(row1,row2)){
// row1 is less than row2
}
使用compare对象就像使用一个普通的compare()函数一样。另外,它的实现可以访问所有存储为成员变量的排序键和排序顺序。
与此方案相同的另一种方法是,把这些排序键和排序顺序存储在全局成员变量中,并且使用一个普通的compare()函数。然而,在全局成员变量之间通信是一种并不提倡的做法,并且可能会产生一些莫名其妙的问题。作为像qStableSort()这样的模板函数的接口,仿函数(函数对象)是一种更为常见的做法。
qStableSort()函数可以接受一个开始迭代器、一个终止迭代器和一个比较函数。这个比较函数是一个带两个参数(两个QStringList)的函数,并且如果第一个参数“小于”第二个参数,它就返回true,否则返回false。传递的作为比较函数的这个compare对象并不是一个真正的函数,但是它可以用作一个函数。