find C++

Keep Open and Learning
Post Reply
星际浪子
Posts: 3597
Joined: 01 May 2009 23:45

find C++

Post by 星际浪子 » 19 Jul 2011 16:06

STL的find,find_if函数提供了一种对数组、STL容器进行查找的方法。使用该函数,需 #include <algorithm>
我们查找一个list中的数据,通常用find(),例如:

using namespace std;
int main()
{
list<int> lst;
lst.push_back(10);
lst.push_back(20);
lst.push_back(30);
list<int>::iterator it = find(lst.begin(), lst.end(), 10); // 查找list中是否有元素“10”
if (it != lst.end()) // 找到了
{
// do something
}
else // 没找到
{
// do something
}
return 0;
}

那么,如果容器里的元素是一个类呢?例如,有list<CPerson> ,其中CPerson类定义如下:

class CPerson
{
public:
CPerson(void);
~CPerson(void);

public:
int age; // 年龄
};

那么如何用find()函数进行查找呢?这时,我们需要提供一个判断两个CPerson对象“相等”的定义,find()函数才能从一个list中找到与指定的CPerson“相等”的元素。
这个“相等”的定义,是通过重载“==”操作符实现的,我们在CPerson类中添加一个方法,定义为:
bool operator==(const CPerson &rhs) const;
实现为:
bool CPerson::operator==(const CPerson &rhs) const
{
return (id == rhs.age);
}

然后我们就可以这样查找(假设list中已经有了若干CPerson对象)了:
list<CPerson> lst;
//////////////////////////////////
// 向lst中添加元素,此处省略
//////////////////////////////////
CPerson cp_to_find; // 要查找的对象
cp_to_find.age = 50;
list<CPerson>::iterator it = find(list.begin(), list.end(), cp_to_find); // 查找

if (it != lst.end()) // 找到了
{
// do something
}
else // 没找到
{
// do something
}
这样就实现了需求。

有人说,如果我有自己定义的“相等”呢?例如,有一个list<CPerson*>,这个list中的每一个元素都是一个对象的指针,我们要在这个list中查找具有指定age的元素,找到的话就得到对象的指针。
这时候,你不再能像上面的例子那样做,我们需要用到find_if函数,并自己指定predicate function(即find_if函数的第三个参数,请查阅STL手册)。先看看find_if函数的定义:
template<class InputIterator, class Predicate>
InputIterator find_if(InputIterator _First, InputIterator _Last, Predicate _Pred);
Parameters
_First
An input iterator addressing the position of the first element in the range to be searched.
_Last
An input iterator addressing the position one past the final element in the range to be searched.
_Pred
User-defined predicate function object that defines the condition to be satisfied by the element being searched for. A predicate takes single argument and returns true or false.


我们在CPerson类外部定义这样一个结构体:
typedef struct finder_t
{
finder_t(int n) : age(n) { } bool operator()(CPerson *p) { return (age == p->age); } int age;
}finder_t;

然后就可以利用find_if函数来查找了:
list<CPerson*> lst;
//////////////////////////////////
// 向lst中添加元素,此处省略
//////////////////////////////////

list<CPerson*>::iterator it = find_if(lst.begin(), lst.end(), finder_t(50)); // 查找年龄为50的人
if (it != lst.end()) // 找到了
{
cout << "Found person with age : " << (*it)->age;
}
else // 没找到
{
// do something
}

星际浪子
Posts: 3597
Joined: 01 May 2009 23:45

Re: find C++

Post by 星际浪子 » 19 Jul 2011 16:06

find_if的第三个参数是个一元函数对象,可以直接定义一个像findById这样的一元谓词,也可以定义一个从binary_function继承下来的二元函数对象,再用绑定器bind1st或者bind2nd转换成一元函数即可。

p.s. 如果id是可动态的就只有用绑定器了,呵呵。

// 二元函数对象
struct findById : binary_function<Mystruct, int, BOOL>
{
BOOL operator()(const Mystruct &_left, int _right) const
{
if (_left.id == _right)
{
return TRUE;
}
else
{
return FALSE;
}
}
};

// 查找
find_if(MyList.begin(), MyList.end(), bind2nd(findById(), 2)));




#include <string>

#include <list>
#include <algorithm>

using namespace std;

struct Mystruct
{
Mystruct(int id)
: m_id(id)
{

}
Mystruct(int id, const char* name)
:m_id(id), m_name(name)
{

}
bool operator() ( Mystruct& st )
{
return m_id==st.m_id;
}
int m_id;
string m_name;
};


int main(int argc, char* argv[])
{
list <Mystruct> MyList;

MyList.push_back(Mystruct(1, "Allan"));
MyList.push_back(Mystruct(2, "Emily"));
MyList.push_back(Mystruct(3, "Andrew"));


list <Mystruct>::iterator iter = std::find_if(MyList.begin(), MyList.end(), Mystruct(2));
if (iter != MyList.end())
{
Mystruct t = *iter;
printf("Found!\n");
cout < <(*iter).m_name < <endl;

}
return 0;
}

Post Reply