目录
一:泛型编程
1. 将程序写得尽可能通用
2. 将算法从特定的数据结构中抽象出来,成为通用的
3. C++的模板为泛型程序设计奠定了关键的基础
二:什么是STL
STL(Standard Template Library),即标准模板库,是一个高效的C++程序库
被容纳于C++标准程序库(C++ Standard Library)中,是ANSI/ISO C++标准中最新的也是极具革命性的一部分
包含了诸多在计算机科学领域里常用的基本数据结构和基本算法。为广大C++程序员们提供了一个可扩展的应用框架,高度体现了软件的可复用性
从逻辑层次来看,在STL中体现了泛型化程序设计的思想(generic programming)
在这种思想里,大部分基本算法被抽象,被泛化,独立于与之对应的数据结构,用于以相同或相近的方式处理各种不同情形
从实现层次看,整个STL是以一种类型参数化(type parameterized)的方式实现的
基于模板(template)
三:STL发展
1971 : David R. Musser 开始倡导Generic Programming 概念
1979 : Alexander Stepanov 创造STL
1987 : Alex 和Musser 开发出一套Ada library
???? : Alex 先后在AT&T 及HP实验室以C 及C++实验大量的体系结构和算法形式。
1992 : Meng Lee 加入称为另一位主要贡献者
1993/11 : Alex 于ANSI/ISO C++ 会议展示
1994 夏: STL 被纳入C++标准
四:STL组件
Container(容器) 各种基本数据结构
Adapter(适配器) 可改变containers或function object接口的一种组件
Algorithm(算法) 各种基本算法如sort、search…等
Iterator(迭代器)* 连接containers和algorithms
Function object(函数对象) *
Allocator(分配器)*
五:容器
容器类是容纳、包含一组元素或元素集合的对象
异类容器类与同类容器类
顺序容器与关联容器
七种基本容器:向量(vector)、双端队列(deque)、链表(list)、集合(set)、多重集合(multiset)、映射(map)和多重映射(multimap)
标准容器的成员绝大部分都具有共同的名称
六:类型成员
七:适配器
适配器是一种接口类
为已有的类提供新的接口
目的是简化、约束、使之安全、隐藏或者改变被修改类提供的服务集合
三种类型的适配器:
容器适配器:用来扩展7种基本容器,它们和顺序容器相结合构成栈、队列和优先队列容器
迭代器适配器
函数对象适配器
八:迭代器
迭代器是面向对象版本的指针,它们提供了访问容器、序列中每个元素的方法
九:算法
C++标准模板库中包括70多个算法
其中包括查找算法,排序算法,消除算法,记数算法,比较算法,变换算法,置换算法和容器管理等等
这些算法的一个最重要的特性就是它们的统一性,并且可以广泛用于不同的对象和内置的数据类型
十:顺序容器
顺序容器的接口
插入方法
push_front(),push_back(),insert(),运算符“=”
删除方法
pop() ,erase(),clear()
迭代访问方法
使用迭代器
其他顺序容器访问方法(不修改访问方法)
front(),back(),下标[]运算符
十一:向量 vector
1. 向量属于顺序容器,用于容纳不定长线性序列(即线性群体),提供对序列的快速随机访问(也称直接访问)
2. 数据结构很像一个数组,所以与其他容器相比,vector 能非常方便和高效访问单个元素,支持随机访问迭代子
3. 向量是动态结构,它的大小不固定,可以在程序运行时增加或减少
与数组不同,向量的内存用尽时,向量自动分配更大的连续内存区,将原先的元素复制到新的内存区,并释放旧的内存区;这是向量类的优点
头文件:#include <vector>
vector 基本操作
(1)头文件
#include<vector>
(2)创建vector对象
vector<int> vec;
(3)尾部插入数字
vec.push_back(a);
(4)使用下标访问元素
cout<<vec[0]<<endl;记住下标是从0开始的
(5)使用迭代器访问元素
vector<int>::iterator it;
for(it=vec.begin();it!=vec.end();it++)
cout<<*it<<endl;
(6)插入元素
vec.insert(vec.begin()+i,a);在第i+1个元素前面插入a;
(7)删除元素
vec.erase(vec.begin()+2);删除第3个元素
vec.erase(vec.begin()+i,vec.end()+j);删除区间[i,j-1];区间从0开始
(8)向量大小
vec.size();
vec.resize;改变大小
(9)清空
vec.clear();
vector,使用示例如下
#include <iostream>
#include <iomanip>
#include <vector> //包含向量容器头文件
using namespace std ;
void main(){
vector<int> A(10); //创建vector对象
int n;
int primecount = 0, i, j;
cout<<"Enter a value>=2 as upper limit: ";
cin >> n;
A[primecount++] = 2;//下标法访问元素
for(i = 3; i < n; i++){
if (primecount == A.size())
A.resize(primecount + 10); //改变容器大小
if (i % 2 == 0)
continue;
j = 3;
while (j <= i/2 && i % j != 0)
j += 2;
if (j > i/2) A[primecount++] = i;
}
for (i = 0; i<primecount; i++){//输出质数
cout<<setw(5)<<A[i];
if ((i+1) % 10 == 0) //每输出10个数换行一次
cout << endl;
}
cout<<endl;
}
十二:双端队列 queue
双端队列是一种放松了访问权限的队列
元素可以从队列的两端入队和出队,也支持通过下标操作符“[]”进行直接访问
与向量的对比:
功能上:和向量没有多少区别,
性能上:在双端队列起点上的插入和删除操作快
头文件:#include <deque>
十三:列表 list
链表主要用于存放双向链表,可以从任意一端开始遍历。链表还提供了拼接(splice)操作,将一个序列中的元素从插入到另一个序列中
对比:
元素的插入和删除操作对 list 而言尤为高效
与 vector 和 deque 相比,对元素的下标访问操作的低效是不能容忍的,因此 list 不提供这类操作
头文件:#include <list>
列表,使用示例如下
#include <iostream>
#include <list>
using namespace std ;
int main(){
list<int> Link; //构造一个列表用于存放整数链表
int i, key, item;
for(i=0;i < 10;i++)// 输入10个整数依次向表头插入{
cin>>item;
Link.push_front(item);
}
cout<<“List: ”; // 输出链表
list<int>::iterator p=Link.begin();
while(p!=Link.end()){ //输出各节点数据,直到链表尾
cout <<*p << " ";
p++; //使P指向下一个节点
}
cout << endl;
cout << "请输入一个需要删除的整数: ";
cin >> key;
Link.remove(key);
cout << "List: "; // 输出链表
p=Link.begin(); // 使P重新指向表头
while(p!=Link.end()){
cout <<*p << " ";
p++; // 使P指向下一个节点
}
cout << endl;
}
十四:关联容器
通过保存在数据项中的索引项,尽可能快速检索数据项
STL标准库中只包含有序关联容器set、multiset、map、multimap
set, multiset:数据项就是索引项; multiset允许出现重复的索引项
map, multimap:数据项是由索引项和其他某种类型的数据组成的一对数据; multimap允许出现重复的索引项
十五:map
1. 增加和删除节点对迭代器的影响很小。对于迭代器来说,可以修改实值,而不能修改key
2. 自动建立Key - value的对应。key 和 value可以是任意你需要的类型
3. 根据key值快速查找记录,查找的复杂度基本是Log(N),如果有1000个记录,最多查找10次,1,000,000个记录,最多查找20次
map的构造函数
1. 使用map得包含map类所在的头文件
#include <map>2. map对象是模板类,需要关键字和存储对象两个模板参数:
map<int, string> personnel;//用int作为索引,存储string对象
map的成员函数
1. map类已经对[]操作符进行了重载
2. 插入2时,先在enumMap中查找主键为2的项,没发现,然后将一个新的对象插入enumMap,键是2,值是一个空字符串,插入完成后,将字符串赋为“Two”;
3. 但是该方法会将每个值都赋为缺省值,然后再赋为显示的值,如果元素是类对象,则开销比较大。可以用以下insert()来避免开销
4. 下标操作符给出了获得一个值的最简单方法:
CString tmp = enumMap[2];
但是,只有当map中有这个键的实例时才对,否则会自动插入一个实例,值为初始化值
5. 我们可以使用Find()和Count()方法来发现一个键是否存在
6. 查找map中是否包含某个关键字条目用find()方法,传入的参数是要查找的key
7. 通过map对象的方法获取的iterator数据类型是一个std::pair对象,包括两个数据 iterator->first 和 iterator->second 分别代表关键字和存储的数据
从map中删除元素
1. 移除某个map中某个条目用erase()
2. 该成员方法的定义如下
iterator erase(iterator it); //通过一个条目对象删除 iterator erase(iterator first, iterator last);//删除一个范围 size_type erase(const Key& key); //通过关键字删除
例如:
enumMap.erase(1);//删掉关键字“1”对应的条目 enumMap.erase(enumMap.begin());//删掉第一个条目 enumMap.erase(enumMap.begin(), enumMap.begin() + 2);//删掉起始的两个条目
3. clear()就相当于
enumMap.erase(enumMap.begin(), enumMap.end());
map,使用示例如下
#include <map>
#include <iostream>
#include <string>
using namespace std;
void main(){
map< string, string > trans_map;
typedef map< string, string >::value_type valType;
trans_map.insert( valType( "001", "grateful" ));
trans_map.insert( valType( "002", "them" ));
trans_map.insert( valType( "003", "because" ));
trans_map.insert( valType( "004", "no" ));
trans_map.insert( valType( "005", "says" ));
trans_map.insert( valType( "006", "thanks" ));
trans_map.insert( valType( "007", "was" ));
trans_map.insert( valType( "008", "suppose" ));
map< string,string >::iterator it;
cout << "Here is our transformation map: \n\n";
for(it=trans_map.begin();it!=trans_map.end();++it)
cout<<"key: "<<(*it).first<<"\t"<<"value: " <<(*it).second<<"\n";
cout<<"Find Key:005"<<endl;
it=trans_map.find("105");
if (it==trans_map.end()){
cout<<"not found"<<endl;
}
else{
cout<<"key: "<<(*it).first <<"\t"<<"value: " <<(*it).second<<"\n";
}
}
十六:multimap
multimap 除了元素对的关键字不是唯一外,与 map 相似
头文件:#include <map>
十七:set
set 可以被视为只有关键字而没有相关的元素值的 map,因此 set 的用户接口也发生了微小的变化:成员类型中没有:
typedef Key value_type;
typedef Cmp value_compare
操作中没有元素的下标访问操作
头文件:#include <set>
十八:multiset
multiset 除了关键字不是唯一外,与 set 相似
头文件:#include <set>
十九:迭代器
迭代器是面向对象版本的指针
指针可以指向内存中的一个地址
迭代器可以指向容器中的一个位置
STL的每一个容器类模版中,都定义了一组对应的迭代器类。使用迭代器,算法函数可以访问容器中指定位置的元素,而无需关心元素的具体类型
迭代器的类型
1. 输入迭代器
可以用来从序列中读取数据
2. 输出迭代器
允许向序列中写入数据
3. 前向迭代器
既是输入迭代器又是输出迭代器,并且可以对序列进行单向的遍历
4. 双向迭代器
与前向迭代器相似,但是在两个方向上都可以对数据遍历
5. 随机访问迭代器
也是双向迭代器,但能够在序列中的任意两个位置之间进行跳转
迭代器的类型表
迭代器,使用示例
#include <list>
#include <iostream>
using namespace std;
int main(){
int i,key;
list<int> intList;
list<int>::iterator it;
cout<<"input 5 digit:";
for(i=0;i<5;i++){
cin>>key;
intList.push_front(key);
}
cout<<"data list:"<<endl;
it=intList.end();
while(1){
cout.width(6);
cout<<*(--it);
if(it==intList.begin())
break;
}
return 0;
}
二十:函数对象
1. 一个行为类似函数的对象,它可以没有参数,也可以带有若干参数,其功能是获取一个值,或者改变操作的状态
2. 任何普通的函数和任何重载了调用运算符operator()的类的对象都满足函数对象的特征
3. STL中也定义了一些标准的函数对象,如果以功能划分,可以分为算术运算、关系运算、逻辑运算三大类;为了调用这些标准函数对象,需要包含头文件<functional>
二十一:已经集成的函数对象
二十二:自定义函数对象
#include <iostream>
using namespace std;
class CFunObj{
public:
void operator()(){
cout<<"hello,function object!"<<endl;
}
int operator()(int i){
cout<<"hello, function object other!"<<endl;
return i+1;
}
private:
int dat;
};
void main(){
CFunObj fo;
fo();
cout<<fo(1)<<endl;
//CFunObj()();
}
二十三:标准C++库中的算法
1. 算法本身是一种函数模板
2. 不可变序列算法(non-mutating algorithms)
不直接修改所操作的容器内容的算法
3. 可变序列算法(mutating algorithms)
可以修改它们所操作的容器的元素
4. 算法部分主要由头文件<algorithm>,<numeric>和<functional>组成
二十四:STL算法的头文件
<algorithm>是所有STL头文件中最大的一个,它是由一大堆模版函数组成的,可以认为每个函数在很大程度上都是独立的,其中常用到的功能范围涉及到比较、交换、查找、遍历操作、复制、修改、移除、反转、排序、合并等等
<numeric>体积很小,只包括几个在序列上面进行简单数学运算的模板函数,包括加法和乘法在序列上的一些操作
<functional>中则定义了一些模板类,用以声明函数对象
二十五:标准函数
二十六:泛型算法例子
#include <iostream>
#include <vector>
#include <algorithm>
#include <iterator>
#include <numeric>
#include <functional>
using namespace std;
int main(){
int ia[] = { 1, 2, 3, 4, 5, 7, 9, 11 };
vector<int> iv(ia, ia+8);
//累加,52
cout<<accumulate(iv.begin(),iv.end(),10)<<endl; //相邻差值
adjacent_difference(iv.begin(),iv.end(),iv.begin());
//复制到ostream_iterator 去, 每列印一个元素, 即加上一个空格
copy(iv.begin(), iv.end(), ostream_iterator<int>(cout, “ ”));
// 1 1 1 1 1 2 2 2
//计算元素值为2 的个数
cout << count(iv.begin(), iv.end(), 2) << endl; // 3
//计算奇数元素的个数
cout << count_if(iv.begin(), iv.end(),
bind2nd(modulus<int>(),2)) << endl; // 5
//从头开始填入新值7, 填3 次
fill_n(iv.begin(), 3, 7);
copy(iv.begin(), iv.end(), ostream_iterator<int>(cout,“ ”));
// 7 7 7 1 1 2 2 2
//内积, 7*7 + 7*7 + 7*7 + 1*1 + 1*1 + 2*2 + 2*2 + 2*2
cout << inner_product(iv.begin(), iv.end(), iv.begin(),0) << endl;
//161
//排序
sort(iv.begin(), iv.end());
copy(iv.begin(), iv.end(), ostream_iterator<int>(cout, “ ”));
// 1 1 2 2 2 7 7 7
//顛倒元素次序
reverse(iv.begin(), iv.end());
copy(iv.begin(), iv.end(), ostream_iterator<int>(cout, “ ”));
// 7 7 7 2 2 2 1 1
//旋转, 交换[first, middle)和[middle, last)
rotate(iv.begin(), iv.begin()+3, iv.begin()+6);
copy(iv.begin(), iv.end(), ostream_iterator<int>(cout, “ ”));
// 2 2 2 7 7 7 1 1
}
二十七:自定义函数作为算法参数
#ifndef CMYCLASS_H
#define CMYCLASS_H
#include <string.h>
class CMyClass{
public:
CMyClass();
CMyClass(string name,int age);
friend bool Less(const CMyClass &num,const CMyClass &myclass);
bool operator==(const CMyClass &myclass);
string &GetName();
int GetAge();
private:
string m_name;
int m_age;};
#endif
CMyClass::CMyClass(){
m_name="";
m_age=12;
}
CMyClass::CMyClass(string name,int age)
{ m_name=name;
m_age=age;
}
bool CMyClass::operator==(const CMyClass &myclass){
return m_name==myclass.m_name;
}
string & CMyClass::GetName(){
return m_name;}
int CMyClass::GetAge(){
return m_age;}
#include "MyClass.h"
#include <iostream>
#include <algorithm>
#include <vector>
#include <functional>
#include <numeric>
using namespace std;
bool Less(const CMyClass &num,const CMyClass &myclass){
return num.m_name<myclass.m_name;
}
void PrintMyClass(CMyClass &myclass){
cout<<"name:"<<myclass.GetName()<<"\t age:"<<myclass.GetAge()<<endl;
}
bool SearchByName(CMyClass &myclass){
return myclass.GetName()=="AAAAZ";
}
void main(){
CMyClass myclass;
vector<CMyClass> vec;
vec.push_back(CMyClass("AAAA",12));
vec.push_back(CMyClass("DFKASDF",12));
vec.push_back(CMyClass("ASDFSAFA",12));
vec.push_back(CMyClass("Z",12));
vec.push_back(CMyClass("AAAAZ",12));
vec.push_back(CMyClass("DFKSADFZ",12));
sort(vec.begin(),vec.end(),Less);
// for (vector<CMyClass>::iterator it=vec.begin();it!=vec.end();it++){
// cout<<it->GetName()<<endl;}
for_each(vec.begin(),vec.end(),PrintMyClass);
vector<CMyClass>::iterator r=find_if(vec.begin(),vec.end(),SearchByName);
cout<<(*r).GetName()<<endl;
}