分类: C/C++
2009-09-19 12:00:38
让我展示些比较酷的东西。构建一个set,比较类型用的是less_equal,然后insert一个10:
set<int, less_equal<int> > s; // s is sorted by “<=”
s.insert(10); // insert the value 10
现在尝试再insert一次10:
s.insert(10);
对于这一个insert的调用,set必须先要搞明白10是否已经位于其中。 我们知道它已经位于其中,但set可是木头木脑的,它必须执行检查。为了便于弄明白发生了什么,我们将已经在set中的10称为
set遍历它的内部数据结构以查找加入10B的位置。 最终,它总要检查10B是否与
!(
好吧,
!(true) && !(true)
再简化就是
false && false
结果当然是false。 也就是说,set得出的结论是
好吧,也许你对酷的定义和我不一样。就算这样,你仍然需要确保你用在关联容器上的比较函数总是对相同的元素返回false。然而,你需要警惕。对这条规则的违反容易到令人吃惊的程度。
举例来说,Item 20 描述了该如何写一个比较函数以使得容纳string *指针的容器根据string的值排序,而不是对指针值排序。那里的比较函数是按升序排序的,但我们现在假设你需要降序排序的比较函数。自然是拿现成的代码来修改了。如果你不细心,可能会这么干,我已经加亮了对Item 20中代码作了改变的部分:
struct StringPtrGreater: // highlights show how
public binary_function
const string*, // from page 89. Beware,
bool> { // this code is flawed!
bool operator()(const string *ps1, const string *ps2) const
{
return !( *ps1 < *ps2); // just negate the old test;
} // this is incorrect!
};
想法是通过将比较函数内部结果取反来反序。很不幸,取反“<”不会给你(你所期望的)“>”,它给你的是“>=”。而你现在知道,因为“>=”将对相同的元素返回true,对关联容器,它不是一个有效的比较函数。
你真正需要的比较类型是这个:
struct StringPtrGreater: // this is a valid
public binary_function
const string*, // associative containers
bool> {
bool operator()(const string *ps1, const string *ps2) const
{
return *ps2 < *ps1; // return whether *ps2
} // precedes *ps1 (i.e., swap
// the order of the
}; // operands)
要避免掉入这个陷阱,你所要记住的就是比较函数的返回值指明的是在此函数定义的排序方式下,一个元素是否应该位于另一个之前。相同的元素绝不该一个领先于另一个,所以比较函数总应该为相同的元素返回false。
唉。
我知道你在想什么。你正在想,“当然,这对set和map很有意义,因为这些容器不能容纳复本。但是multiset和multimap怎么样呢?那些容器可以容纳复本,那些容器可能包含副本,因此,如果容器认为两个相同元素对象不等值,我需要注意些什么(so what do I care if the container thinks that two objects of equal value aren’t equivalent?)?它将会把两个都存储进去的,这正是multi系列容器的所要支持的事情。没有问题存在,对吧?”
错。想知道为什么,让我们回过头去看最初的例子,但这次使用的是一个mulitset:
multiset<int, less_equal<int> > s; // s is still sorted by “<=”
s.insert(10); // insert
s.insert(10); // insert 10 B
现在,s里面有二个10的元素拷贝,因此我们期望,如果我们在它上面做一个equal_range,我们将会得到一对iterator以指出包含这两个拷贝的范围。但那是不可能的。equal_range,名不符实,不是指示出相同的元素的范围,而是等值的元素的范围。在这个例子中,s的比较函数说
你明白了吗?除非你的比较函数总是为相同的元素返回false,你将会打破所有的关联型容器,不管它们是否允许存储复本。
从技术上讲,用于关联容器的比较函数必须在它们所比较的对象上定义一个“strict weak ordering”。 ( 传给sort(Item 31)等泛型算法的比较函数也有同样的限制)。如果你对strict weak ordering的含义细节感兴趣,可在很多STL指导书籍中找到,比如Josuttis的《The C++ Standard Library》(WQ注:中文版P176,英文版电子P156),Austern的《Generic Programming and the STL 》,和SGI STL的网站。 我从未发现这个细节如此重要,但一个对strict weak ordering的要求直接瞄上了这个Item。 那个要求就是任何定义了一个strict weak ordering 的函数都必须在传入相同元素的两个拷贝时返回false。
嗨! 那是这个Item!