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——泛型编程的更多相关文章

  1. C++ STL泛型编程——在ACM中的运用

    学习过C++的朋友们应该对STL和泛型编程这两个名词不会陌生.两者之间的关系不言而喻,泛型编程的思想促使了STL的诞生,而STL则很好地体现了泛型编程这种思想.这次想简单说一下STL在ACM中的一些应 ...

  2. C++ -- STL泛型编程(一)之vector

    STL提供三种组件:容器,迭代器,算法,它们都支持泛型程序设计标准容器有两类:顺序容器和关联容器. 顺序容器(vector,list,deque,string等)是一系列元素的有序组合. 关联容器(s ...

  3. C++ -- STL泛型编程(二)之set

    set集合容器实现了红黑树的平衡二叉检索树的数据结构,在插入元素时候它会自动调整二叉树的排列,把元素放在适当的位置,以确保每个子树根节点的键值都大于左子树的所有节点的键值,而小于右子树的所有节点的键值 ...

  4. [GeekBand] STL与泛型编程(1)

    在C++语法的学习过程中,我们已经对模板有了基本的了解.泛型编程就是以模板为工具的.泛化的编程思想.本篇文章介绍了一些在之前的文章中没有涉及到的一些模板知识.泛型编程知识和几种容器.关于模板的一些重复 ...

  5. 泛型编程、STL的概念、STL模板思想及其六大组件的关系,以及泛型编程(GP)、STL、面向对象编程(OOP)、C++之间的关系

    2013-08-11 10:46:39 介绍STL模板的书,有两本比较经典: 一本是<Generic Programming and the STL>,中文翻译为<泛型编程与STL& ...

  6. STL之父Stepanov谈泛型编程的发展史

    这是一篇Dr. Dobb's Journal对STL之父stepanov的采访.文中数次提到STL的基本思想.语言的特性.编程的一些根本问题等,非常精彩.这篇文章让我想去拜读下stepanov的大作& ...

  7. [GeekBand] STL与泛型编程(2)

    本篇文章在上一篇文章的基础上进一步介绍一些常用的容器以及STL的一些深入知识. 一. Stack和Queue 栈和队列是非常常用的两种数据结构,由deque适配而来.关于数据结构的知识这里就不在介绍了 ...

  8. [GeekBand] STL与泛型编程(3)

    本篇文章主要介绍泛型算法中的变易.排序.数值算法. 一. 变易算法 所谓变易算法是指那些改变容器中的对象的操作. 1.1 copy组 template <class InputIterator, ...

  9. STL与泛型编程(第一周)

    part 1 C++模版简介 一,模版概观 1.模板 (Templates)是C++的一种特性,允许函数或类(对象)通过泛型(generic types)的形式表现或运行. 模板可以使得函数或类在对应 ...

随机推荐

  1. solidity语言11

    函数修饰符 pragma solidity ^0.4.11; contract owned { address owner; // 构造函数 function owned() public { own ...

  2. BZOJ4566:[HAOI2016]找相同字符(SAM)

    Description 给定两个字符串,求出在两个字符串中各取出一个子串使得这两个子串相同的方案数.两个方案不同当且仅当这两 个子串中有一个位置不同. Input 两行,两个字符串s1,s2,长度分别 ...

  3. Codeforces Round #333 (Div. 1)

    A. The Two Routes In Absurdistan, there are n towns (numbered 1 through n) and m bidirectional railw ...

  4. [19/03/17-星期日] 常用类_Calendar日历类&GregorianCalendar公历日历类

    一.概念 Calendar 类是一个抽象类,为我们提供了关于日期计算的相关功能,比如:年.月.日.时.分.秒的展示和计算. GregorianCalendar 是 Calendar 的一个具体子类,提 ...

  5. 2018.10.4 AndroidStudio

    AndroidStudio低版本sdkversion开发 Error:Minimum supported Gradle version is 4.1 Current version is 2.14.1 ...

  6. $_GET 与 $POST

    $_GET就是地址传值,用 '?' 开始传值,多个值间用 '&' 号分隔,多用于简单的传值,比如说看新闻需要新闻id一般就会用地址传值, $_GET的好处是传值可见,也就是只要一个地址就ok了 ...

  7. IOS异步获取数据并刷新界面dispatch_async的使用方法

    在ios的开发和学习中多线程编程是必须会遇到并用到的.在java中以及Android开发中,大量的后台运行,异步消息队列,基本都是运用了多线程来实现. 同样在,在ios移动开发和Android基本是很 ...

  8. djano-模板层基础知识

    ########模板层######## 模板层其实就是templates文件夹里的html文件 其实这里的每个html不是真正意义的上html代码,只有经过模板渲染过后才算的上真正的html页面. 一 ...

  9. logback.xml模板详解

    <?xml version="1.0" encoding="UTF-8"?> <!-- 配置文件每隔1分钟,就检查更新 --> < ...

  10. 自定义组件v-model的实质性理解

    用了几个月Vue一直很纠结自定义组件的v-model实现,最近开始学习React时,React中受控组件与状态提升的理念与v-model不谋而合. 转载请注明地址: https://www.cnblo ...