[C++ STL] 一个泛型算法的设计
2013-03-12 16:26
471 查看
从一个简单的需求开始,从一个vector中返回小于10的数的vector,那我们会写如下的函数:
vector<int> less_than_ten(vector<int>& vec) {
vector<int> result;
for(int i=0; i<vec.size(); i++) {
if (vec[i] < 10) {
result.push_back(vec[i]);
}
}
return result;
}
这个函数只局限于能返回小于10的数,我想能够返回小于我指定的数,于是改成了如下的函数:
vector<int> less_than(vector<int>& vec, int val) {
vector<int> result;
for (int i=0; i < vec[i]; i++) {
if (vec[i] < val)
result.push_back(vec[i]);
}
return result;
}
然后我又考虑到,不应该仅仅是小于,还有可能是大于,或等于。参数化比较函数,于是有了如下的版本 :
bool less_than(int x, int y) {
return x < y;
}
bool greater_than(int x, int y) {
return x > y;
}
vector<int> filter(vector<int>& vec, int val, bool (*pred)(int,int)) {
vector<int> result;
for (int i=0; i < vec.size(); i++) {
if (pred(vec[i],val))
result.push_back(vec[i]);
}
return result;
}
不能仅仅只是int型,如果我们还要float,long,char型 难道都要写一个吗?,不需要,只要使用模板就可以了:
template<typename elemType>
vector<elemType> filter(vector<elemType>& vec, elemType val, bool (*pred) (elemType,elemType)) {
vector<elemType> result;
for (int i=0; i < vec.size(); i++) {
if (pred(vec[i],val))
result.push_back(vec[i]);
}
return result;
}
此时需要注意的是调用的时候,必须保证这三个参数的类型要一致。
在广泛一点我们不仅仅是想使用vector容器,也可以用别的容器,返回的是一个iterator 指向容器中一个元素。
我们使用C++库<algorithm>中的提供的函数对象,和泛型算法.
C++ algorithm库里面提供的 function object 有plus<>,minus<>等等。
参考http://www.cplusplus.com/reference/std/functional/plus/
我们可以使用bind2nd 把它绑定到一个固定的值,与一个固定值进行比较
参考:http://www.cplusplus.com/reference/std/functional/bind2nd/
template<typename elemType,typename Com>
vector<elemType> filter(vector<elemType>& vec, elemType val, Com pred) {
vector<elemType> result;
vector<elemType>::iterator it;
for(; (it = find_if(vec.begin().vec.end(),bind2nd(pred,val))) != vec.end(); it++)
result.push_back(*it);
return result;
}
find_if是在一个范围内查找符合条件的数,并返回该数的位置。 找到最后就是vec.end()了。
for(; (it = find_if(vec.begin().vec.end(),bind2nd(pred,val))) != vec.end(); it++) 这个就是遍历从vec 找到所有符合pred 这个比较函数的值,并把它添加到result里,然后返回。
我们没有必要一开始就能立即写出泛型的函数,可以从简单的开始,一步一步抽象,并参数化。每次写完一个函数的时候,我们应该思考它的扩展性,看能否使其更灵活。这样做非常有助于我们的提高。好的代码要写两次。
参考书目:
《Essential C++》
vector<int> less_than_ten(vector<int>& vec) {
vector<int> result;
for(int i=0; i<vec.size(); i++) {
if (vec[i] < 10) {
result.push_back(vec[i]);
}
}
return result;
}
这个函数只局限于能返回小于10的数,我想能够返回小于我指定的数,于是改成了如下的函数:
vector<int> less_than(vector<int>& vec, int val) {
vector<int> result;
for (int i=0; i < vec[i]; i++) {
if (vec[i] < val)
result.push_back(vec[i]);
}
return result;
}
然后我又考虑到,不应该仅仅是小于,还有可能是大于,或等于。参数化比较函数,于是有了如下的版本 :
bool less_than(int x, int y) {
return x < y;
}
bool greater_than(int x, int y) {
return x > y;
}
vector<int> filter(vector<int>& vec, int val, bool (*pred)(int,int)) {
vector<int> result;
for (int i=0; i < vec.size(); i++) {
if (pred(vec[i],val))
result.push_back(vec[i]);
}
return result;
}
不能仅仅只是int型,如果我们还要float,long,char型 难道都要写一个吗?,不需要,只要使用模板就可以了:
template<typename elemType>
vector<elemType> filter(vector<elemType>& vec, elemType val, bool (*pred) (elemType,elemType)) {
vector<elemType> result;
for (int i=0; i < vec.size(); i++) {
if (pred(vec[i],val))
result.push_back(vec[i]);
}
return result;
}
此时需要注意的是调用的时候,必须保证这三个参数的类型要一致。
在广泛一点我们不仅仅是想使用vector容器,也可以用别的容器,返回的是一个iterator 指向容器中一个元素。
我们使用C++库<algorithm>中的提供的函数对象,和泛型算法.
C++ algorithm库里面提供的 function object 有plus<>,minus<>等等。
参考http://www.cplusplus.com/reference/std/functional/plus/
我们可以使用bind2nd 把它绑定到一个固定的值,与一个固定值进行比较
参考:http://www.cplusplus.com/reference/std/functional/bind2nd/
template<typename elemType,typename Com>
vector<elemType> filter(vector<elemType>& vec, elemType val, Com pred) {
vector<elemType> result;
vector<elemType>::iterator it;
for(; (it = find_if(vec.begin().vec.end(),bind2nd(pred,val))) != vec.end(); it++)
result.push_back(*it);
return result;
}
find_if是在一个范围内查找符合条件的数,并返回该数的位置。 找到最后就是vec.end()了。
for(; (it = find_if(vec.begin().vec.end(),bind2nd(pred,val))) != vec.end(); it++) 这个就是遍历从vec 找到所有符合pred 这个比较函数的值,并把它添加到result里,然后返回。
我们没有必要一开始就能立即写出泛型的函数,可以从简单的开始,一步一步抽象,并参数化。每次写完一个函数的时候,我们应该思考它的扩展性,看能否使其更灵活。这样做非常有助于我们的提高。好的代码要写两次。
参考书目:
《Essential C++》
相关文章推荐
- [C++ STL] 一个泛型算法的设计
- C++ 之 高效使用STL(泛型算法设计原理解析)
- 设计一个泛型算法
- C++STL总结(附带容器迭代器泛型算法详情链接【未完成】)
- <学习笔记>Algorithm Library Design 算法库设计in c++ II(STL与泛型编程)
- C++ Primer 学习笔记_45_STL实践与分析(19)--泛型算法的结构
- [学习]C++ STL(自定义泛型算法)
- 一个泛型算法的设计
- 如何设计一个泛型算法
- 2017 程序设计实习之C++部分作业题汇总 - H:STL 容器与算法
- 用C++设计一个不能被继承的类
- 请设计一个算法,给一个字符串进行二进制编码,使得编码后字符串的长度最短。(哈夫曼树)
- STL泛型算法总结
- C++ STL 排列算法
- [转贴]从零开始学C++之STL(二):实现一个简单容器模板类Vec(模仿VC6.0 中 vector 的实现、vector 的容量capacity 增长问题)
- 用C++设计一个不能被继承的类
- C++泛型算法
- 二叉排序树中,令f = (最大值+最小值) / 2,设计一个算法, 找出距离f值最近、大于f值的结点。复杂度不能为O(n2)。
- 最近准备做一个C++模式设计方面的学习总结,希望有路过看见的人推荐一下
- 一个文本情感识别与舆情分析的算法设计思路