STL——泛型编程
1、指针的算术运算
对于一个存储int数据的vector,我们要查找某值是否存在于其中,采用下标操作的做法如下:
int* find(const vector<int> &vec, int value)
{
for(int i = 0; i < vec.size(); i++)
{
if(vec[i] == value)
return &vec[i];
}
return 0;
}
此函数对于存储int型数据的vector实用,但是如果对于其他类型的数据呢?为了达到这个目的,我们使用function template的形式:
template <typename elemType>
elemType* find(const vector<elemType> &vec, const elemType &value)
{
for(int i = 0; i < vec.size(); i++)
{
if(vec[i] == value)
return &vec[i];
}
return 0;
}
现在,新任务出现了,我们需要让这样一个函数能处理vector和array的元素。对于这个问题,我们的解决办法是,传参容器中的元素而不是指明的容器。
对于array,要传入容器中的元素,可以使用array的首尾元素的指针,即:
template <typename elemType>
elemType* find(const elemType *first, const elemType *last, const elemType &value)
{
if(!first || !last)
return 0;
for(;first != last; first++)
{
if(*first == value)
return first;
}
return 0;
}
对于vector,由于其为连续存储结构,同样可以使用此方式,传入容器的首尾元素指针:
vector<string> svec;
find(&svec[0], &svec[svec.size()], search_value); //当然,首先得保证vector不为空,这里不做多的描述
到此为止,我们可以使用同样的find函数来查找vector和array中的指定元素了,但是此时,又有个新任务,需要find支持list的元素查找。因为list不是连续存储的,所以在find函数体内,就不能使用指针的++运算来获取下一个元素的地址了。
为了解决这个问题,我们在指针的行为之上提供一层抽象化机制,这个机制可以屏蔽掉vector和list的指针操作行为,使用户在find中看不到指针的操作,从而看到vector和list的操作一样,这样就可以将find函数应用于所有的容器类。
2、Iterators(泛型指针)
这层抽象化机制,即iterator类,跟指针的操作类似,主要提供运算有:++,*,==,!=
定义这样一个类,我们需要知道两个信息
(1)迭代对象类型(即容器类型),这可以决定怎么存取下一个元素
(2)迭代器指向的元素类型,决定*(取值,类似指针的*)操作的返回值
例如:
vector<string>::iterator iter; //iter为vector容器类型的迭代器,指向的元素类型为string类型。
“;;”符号标识迭代器为容器类中的嵌套类型。
那么现在的find函数为:
template <typename IteratorType, typename elemType>
elemType* find(IteratorType first, IteratorType last, const elemType &value)
{
for(;first != last; first++)
{
if(*first == value)
return first;
}
return last;
}
此版本的find函数还不够弹性,原因在于,传入的elemType类型value并不一定支持“==”运算。这个问题的解决办法在于,传入一个函数指针,利用这个函数指针调用的函数来判断,从而取代“==”。这种方法将在后面介绍。
3、泛型算法函数
头文件:#include<algorithm>
搜索算法:find(),count(),adjacent_find(),find_if(),count_if(),binary_search(),find_first_of().
排序算法:merge(),partial_sort(),partition(),random_shuffle(),reverse(),rotate(),sort().
复制删除替换算法:copy(),remove(),remove_if(),replace(),replace_if(),swap(),unique().
关系算法:equal(),includes(),mismatch().
生成与质变算法:fill(),for_each(),generate(),transform().
数值算法:accmulate(),adjacent_difference(),partial_sum(),inner_product().
集合算法:set_union(),set_difference().
4、设计一个泛型算法
需求:找出一组数据中小于10的数
首先想到的算法就是将这组数据中的每个数与10对比,然后满足条件的存储起来。然而,如果需求改变为小于11或者大于10,那么原来的函数就不能用了,所以考虑到这个问题,实现如下算法:
bool less_than(int num1, int num2)
{
return num1 < num2 ? True : False;
}
bool greater_than(int num1, int num2)
{
return num1 > num2 ? True : False;
}
vector<int> filter(const vector<int> &vec, int filter_value, bool (*judge)(int, int))
{
vector<int> vecRe;
vector<int>::iterator itBegin = vec.begin();
vector<int>::iterator itEnd = vec.end();
for(;itBegin != itEnd; itBegin++)
{
if(judge(*itBegin, filter_value))
{
vecRe.push_back(*itBegin);
}
}
return vecRe;
}
当然,还可以使用template实现更加泛型的方法,并且使用function object替代函数指针来提高效率,这里就不讨论了。
STL——泛型编程的更多相关文章
- C++ STL泛型编程——在ACM中的运用
学习过C++的朋友们应该对STL和泛型编程这两个名词不会陌生.两者之间的关系不言而喻,泛型编程的思想促使了STL的诞生,而STL则很好地体现了泛型编程这种思想.这次想简单说一下STL在ACM中的一些应 ...
- C++ -- STL泛型编程(一)之vector
STL提供三种组件:容器,迭代器,算法,它们都支持泛型程序设计标准容器有两类:顺序容器和关联容器. 顺序容器(vector,list,deque,string等)是一系列元素的有序组合. 关联容器(s ...
- C++ -- STL泛型编程(二)之set
set集合容器实现了红黑树的平衡二叉检索树的数据结构,在插入元素时候它会自动调整二叉树的排列,把元素放在适当的位置,以确保每个子树根节点的键值都大于左子树的所有节点的键值,而小于右子树的所有节点的键值 ...
- [GeekBand] STL与泛型编程(1)
在C++语法的学习过程中,我们已经对模板有了基本的了解.泛型编程就是以模板为工具的.泛化的编程思想.本篇文章介绍了一些在之前的文章中没有涉及到的一些模板知识.泛型编程知识和几种容器.关于模板的一些重复 ...
- 泛型编程、STL的概念、STL模板思想及其六大组件的关系,以及泛型编程(GP)、STL、面向对象编程(OOP)、C++之间的关系
2013-08-11 10:46:39 介绍STL模板的书,有两本比较经典: 一本是<Generic Programming and the STL>,中文翻译为<泛型编程与STL& ...
- STL之父Stepanov谈泛型编程的发展史
这是一篇Dr. Dobb's Journal对STL之父stepanov的采访.文中数次提到STL的基本思想.语言的特性.编程的一些根本问题等,非常精彩.这篇文章让我想去拜读下stepanov的大作& ...
- [GeekBand] STL与泛型编程(2)
本篇文章在上一篇文章的基础上进一步介绍一些常用的容器以及STL的一些深入知识. 一. Stack和Queue 栈和队列是非常常用的两种数据结构,由deque适配而来.关于数据结构的知识这里就不在介绍了 ...
- [GeekBand] STL与泛型编程(3)
本篇文章主要介绍泛型算法中的变易.排序.数值算法. 一. 变易算法 所谓变易算法是指那些改变容器中的对象的操作. 1.1 copy组 template <class InputIterator, ...
- STL与泛型编程(第一周)
part 1 C++模版简介 一,模版概观 1.模板 (Templates)是C++的一种特性,允许函数或类(对象)通过泛型(generic types)的形式表现或运行. 模板可以使得函数或类在对应 ...
随机推荐
- ubuntu下go开发环境
https://qiita.com/necomeshi/items/676ccb669d6e6102117b 安装 https://golang.org/dl/ # 下载&解压 axel -n ...
- jdk是什么?jdk1.8安装配置方法
jdk是什么呢?jdk的是java development kit的缩写,意思是java程序开发的工具包.也可以说jdk是java的sdk. 目前的JDK大致分三个大版本:Java SE:Java P ...
- Uva 12298 超级扑克2
题目链接:https://vjudge.net/problem/UVA-12298 题意: 1.超级扑克,每种花色有无数张牌,但是,这些牌都是合数:比如黑桃:4,6,8,9,10,,,, 2.现在拿走 ...
- Several ports (8005, 8080, 8009) required by Tomcat v8.5 Server at localhost are already in use. The server may already be running in another process, or a system process may be using the port. To sta
eclipse出现:Several ports (8005, 8080, 8009) required by Tomcat v8.5 Server at localhost are already i ...
- JavaEE权限管理系统的搭建(六)--------使用拦截器实现菜单URL的跳转权限验证和页面的三级菜单权限按钮显示
本小结讲解,点击菜单进行页面跳转,看下图,点击管理员列表后会被认证拦截器首先拦截,验证用户是否登录,如果登录就放行,紧接着会被权限验证拦截器再次拦截,拦截的时候,会根据URL地址上找到对应的方法,然后 ...
- ProjectServer如何创建时间表
默认配置的ProjectServer是没有时间表的,任务汇报的时候不能汇报工时,只能汇报任务的百分比. 但如果有企业一定要用工时来汇报的话,我们就需要开启时间表. 点击服务器设置-->时间报告阶 ...
- AutoCAD笔记
1.等距离复制:使用菜单栏下“修改->阵列”功能或右侧快捷方式,可以轻松复制,以下是阵列设置界面 2.快捷键-图案填充和渐变色:H+空格 3.画图时时常会用到Ctrl+V,但难免会按成了Ctrl ...
- Question 20171117 Java中的编码问题?
撰文缘由 前几天做一个邮件发送功能,一些常用信息配置在properties文件中,通过prop.getProperty(key)来获取配置的信息,结果配置文件中是用中文写的,邮件发送成功后,邮箱中的激 ...
- cornerstone提交报错"but is missing"以及xocde提示"missing from working copy"
问题描述 xocde提示"missing from working copy" 虽然这种警告不会影响程序到运行,但是数量很多,而且在svn提交的时候回出现这种问题 使用的svn工具 ...
- UICollectionViewCell的设置间距
UICollectionViewCell的设置间距 #pragma mark - UICollectionView 大小(宽高,平均一行三个) - (CGSize)collectionView:(UI ...