STL(Standard Template Library,标准模板库)
STL的三大组件:容器(container)、算法(algorithm)、迭代器(iterator)。
算法操作数据,容器存储数据,迭代器是算法操作容器的桥梁,迭代器和容器一一对应。
STL六大组件
容器 算法 迭代器 仿函数 适配器 空间配置器
容器:存放数据
算法:操作数据
迭代器:容器和算法的桥梁
仿函数:为算法提供更多的策略
适配器:为算法提供更多的参数接口
空间配置器:管理容器和算法的空间
算法的分类:
质变算法:是指运算过程中会改变区间元素的内容。例如拷贝,替换,删除等。
非质变算法:是指运算过程中不会区间的元素内容,例如查找、计数、遍历、寻找极值。
迭代器的分类
案例:容器vector
#include
#include
#include
void myPrintInt(int num);
using namespace std;
void test01(){
//单端动态数组vector类模板
vector<int> arr(0);
//push_back()尾部插入数据
arr.push_back(100);
arr.push_back(200);
arr.push_back(300);
arr.push_back(400);
//访问数据
//定义一个迭代器存储arr的起始迭代器
vector<int>::iterator beginIt = arr.begin();
//定义一个迭代器存储arr的结束迭代器
vector<int>::iterator endIt = arr.end();
//for循环遍历1
for(vector<int>::iterator i = beginIt; i != endIt; i++){
//对迭代器取* 代表的是 容器的元素
//*biginIt
cout << *i << " ";
}
cout << endl;
//for循环遍历2(推荐)
for(vector<int>::iterator beginIt = arr.begin(); beginIt != arr.end();beginIt++){
cout << *beginIt << " ";
}
cout << endl;
//STL提供的算法来遍历容器(包含算法头文件algorithm)
//for_each从容器的起始--->结束,逐个元素取出
//myPrintInt容器数据的打印方式
for_each(arr.begin(),arr.end(),myPrintInt);
}
void myPrintInt(int num){
cout << num << " ";
}
int main(int argc, char *argv[])
{
test01();
return 0;
}
运行结果:
案例2:容器存放自定义数据类型
#include
#include
#include
#include
using namespace std;
class Person{
friend void myPrintInt1(Person &ob);
private:
string name;
int age;
public:
Person(string name,int age){
this->name = name;
this->age = age;
}
};
void myPrintInt1(Person &ob){
cout << ob.age << " " << ob.name << endl;
}
void test02(){
vector<Person> arr;
Person p1("tom",15);
arr.push_back(p1);
arr.push_back(Person("davi",16));
arr.push_back(Person("mary",17));
arr.push_back(Person("peter",18));
for_each(arr.begin(),arr.end(),myPrintInt1);
}
int main(int argc, char *argv[])
{
test02();
return 0;
}
运行结果:
案例3:容器嵌套容器
void test03(){
vector<int> v1;
vector<int> v2;
vector<int> v3;
v1.push_back(10);
v1.push_back(20);
v1.push_back(30);
v1.push_back(40);
v2.push_back(100);
v2.push_back(200);
v2.push_back(300);
v2.push_back(400);
v3.push_back(1000);
v3.push_back(2000);
v3.push_back(3000);
v3.push_back(4000);
vector< vector<int>> v4;
v4.push_back(v1);
v4.push_back(v2);
v4.push_back(v3);
for(vector<vector<int>>::iterator it = v4.begin(); it != v4.end(); it++){
for(vector<int>::iterator mit = (*it).begin(); mit != (*it).end();mit++){
cout << (*mit) << " ";
}
cout << endl;
}
}
运行结果:
#include
#include
using namespace std;
/*
3.1.2.1 string 构造函数
string();//创建一个空的字符串 例如: string str;
string(const string& str);//使用一个 string 对象初始化另一个 string 对象
string(const char* s);//使用字符串 s 初始化
string(int n, char c);//使用 n 个字符 c 初始化
3.1.2.2 string 基本赋值操作
string& operator=(const char* s);//char*类型字符串 赋值给当前的字符串
string& operator=(const string &s);//把字符串 s 赋给当前的字符串
string& operator=(char c);//字符赋值给当前的字符串
string& assign(const char *s);//把字符串 s 赋给当前的字符串
string& assign(const char *s, int n);//把字符串 s 的前 n 个字符赋给当前的字符串
string& assign(const string &s);//把字符串 s 赋给当前字符串
string& assign(int n, char c);//用 n 个字符 c 赋给当前字符串
string& assign(const string &s, int start, int n);//将 s 从 start 开始 n 个 字符赋值给字符串
*/
void test01(){
//string(const char* s);//使用字符串s初始化
string str1 = "hello";
cout << str1 << endl;
//string(int n, char c);//使用n个字符c初始化
string str2(10,'H');
cout<<str2<<endl;//"HHHHHHHHHH"
string str3 = str2;
cout << str3 << endl;//"HHHHHHHHHH"
string str4;
//string& operator=(const string &s);//把字符串s赋给当前的字符串
str4 = str1;
cout << str4 << endl;//hello
//string& operator=(const char* s);//char*类型字符串 赋值给当前的字符串
string str5;
str5 = "hello str5";
cout << str5 << endl; //hello str5
//string& operator=(char c);//字符赋值给当前的字符串
string str6;
str6 = 'H';
cout << str6 << endl;//"H"
//string& assign(const char *s);//把字符串s赋给当前的字符串
string str7 = "dsds";
str7.assign("hello str7");
cout<<str7<<endl;//"hello str7"
//string& assign(const char *s, int n);//把字符串s的前n个字符赋给当前的字符串
string str8;
str8.assign("hello str8",5);
cout << str8 << endl;
//string& assign(const string &s);//把字符串s赋给当前字符串
string str9;
str9.assign(str8);
cout << str8 << endl;//"hello"
//string& assign(int n, char c);//用n个字符c赋给当前字符串
string str10;
str10.assign(10,'w');
cout << str10 << endl;//"wwwwwwwwww"
//string& assign(const string &s, int start, int n);//将s从start开始n个赋给当前字符串
string str11;
str11.assign("hehehahahaxixi", 4, 6);
cout<<str11<<endl;//"hahaha"
}
int main(int argc, char *argv[])
{
test01();
return 0;
}
void test02(){
string str1="hello string";
cout<<str1[1]<<endl;//'e'
cout<<str1.at(1)<<endl;//'e'
str1[1]='E';
cout<<str1<<endl;//"hEllo string"
str1.at(7) = 'T';
cout<<str1<<endl;//"hEllo sTring"
//[]和at的区别
try{
//str1[1000]='G';//越界 []不抛出异常
str1.at(1000)='G';//越界 at会抛出异常
}catch(exception &e){
cout<<"异常:"<<e.what()<<endl;
}
}
运行结果:
/*
* 3.1.2.4 string拼接操作
* string& operator+=(const string& str);//重载+=操作符
* string& operator+=(const char* str);//重载+=操作符
* string& operator+=(const char c);//重载+=操作符
* string& append(const char *s);//把字符串s连接到当前字符串结尾
* string& append(const char *s, int n);//把字符串s的前n个字符连接到当前字符串 结尾
* string& append(const string &s);//同operator+=()
* string& append(const string &s, int pos, int n);//把字符串s中从pos开始的n个 字符连接到当前字符串结尾
* string& append(int n, char c);//在当前字符串结尾添加n个字符c
*/
void test03(){
string str1="hello";
string str2=" string";
//string& operator+=(const string& str);//重载+=操作符
str1 += str2;
cout<<str1<<endl;//"hello string"
string str3="hello";
//string& operator+=(const char* str);//重载+=操作符
str3 += " string";
cout<<str3<<endl;//"hello string"
string str4="hello";
//string& append(const char *s, int n);//把字符串s的前n个字符连接到当前字 符串结尾
str4.append("hehehaha",4);
cout<<str4<<endl;//"hellohehe"
//string& append(const string &s, int pos, int n);//把字符串s中从pos开始 的n个字符连接到当前字符串结尾
string str5="hello";
string str6="hehehahaha";
str5.append(str6,4,6);
cout<<str6<<endl;//"hellohahaha"
}
int main(int argc, char *argv[])
{
test03();
return 0;
}
运行结果:
/*
* 3.1.2.5 string查找和替换
* int find(const string& str, int pos = 0) const; //查找str第一次出现位置,从p os开始查找
* int find(const char* s, int pos = 0) const; //查找s第一次出现位置,从pos开始 查找
* int find(const char* s, int pos, int n) const; //从pos位置查找s的前n个字符 第一次位置
* int find(const char c, int pos = 0) const; //查找字符c第一次出现位置
* int rfind(const string& str, int pos = npos) const;//查找str最后一次位置,从 pos开始查找
* int rfind(const char* s, int pos = npos) const;//查找s最后一次出现位置,从po s开始查找
* int rfind(const char* s, int pos, int n) const;//从pos查找s的前n个字符最后 一次位置
* int rfind(const char c, int pos = 0) const; //查找字符c最后一次出现位置
* string& replace(int pos, int n, const string& str); //替换从pos开始n个字符 为字符串str
* string& replace(int pos, int n, const char* s); //替换从pos开始的n个字符为 字符串s
*
*/
void test04(){
//int find(const string& str, int pos = 0) const; //查找str第一次出现位 置,从pos开始查找
string str1="hehe:haha:xixi:haha:heihei";
//从str1中找haha
string tmp="haha";
cout<<str1.find(tmp)<<endl;//5
cout<<str1.find(tmp,10)<<endl;//15
//int find(const char* s, int pos = 0) const; //查找s第一次出现位置,从pos 开始查找
cout<<str1.find("haha")<<endl;//5
str1.replace(5,4,"###");
cout<<str1<<endl;//"hehe:###:xixi:haha:heihei"
string str2="www.sex.117114.sex.person.77.com";
//需求:将字符串中的所有"sex"用***屏蔽
int ret = 0;
while((ret = str2.find("sex")) < str2.size()){
str2.replace(ret,strlen("sex"),"***");
}
cout<<str2<<endl;
}
int main(int argc, char *argv[])
{
test04();
return 0;
}
运行结果:
/*
* compare函数在>时返回 1,<时返回 ‐1,==时返回 0。
* 比较区分大小写,比较时参考字典顺序,排越前面的越小。
* 大写的A比小写的a小。
* int compare(const string &s) const;//与字符串s比较
* int compare(const char *s) const;//与字符串s比较
*/
void test05(){
string str1="hehe";
string str2 = "haha";
cout<<str1.compare(str2)<<endl;//1
cout<<str1.compare("lala")<<endl;//‐1
cout<<str1.compare("hehe")<<endl;//0
}
int main(int argc, char *argv[])
{
test05();
return 0;
}
运行结果:
/*
* string substr(int pos = 0, int n = npos) const;//返回由pos开始的n个字符组成 的字符串
*/
void test06(){
string str1="hehehe:ha:xixixi:lalala:heihei";
//cout<
//案例:将分割的所有字符串提取出来
int pos = 0;
while(1){
int ret = str1.find(":",pos);
if(ret < 0){
string tmp = str1.substr(pos,(str1.size() - pos));
cout<<tmp<<endl;
break;
}
string tmp = str1.substr(pos,ret-pos);
cout<< tmp <<endl;
pos = ret+1;
}
}
int main(int argc, char *argv[])
{
test06();
return 0;
}
运行结果:
/*
* string& insert(int pos, const char* s); //插入字符串
* string& insert(int pos, const string& str); //插入字符串
* string& insert(int pos, int n, char c);//在指定位置插入n个字符c
* string& erase(int pos, int n = npos);//删除从Pos开始的n个字符
*
*/
void test07(){
string str1="hello world";
str1.insert(5,"hehe");
cout<<str1<<endl;//"hellohehe world
str1.erase(5,4);//删除字符串中hehe
cout<<str1<<endl;//"hello world"
//清空字符串 str1.size()得到字符串的总大小
str1.erase(0,str1.size());
cout<<str1.size()<<endl;//0
}
int main(int argc, char *argv[])
{
test07();
return 0;
}
运行结果:
void test08(){
string str1;//对象
char *str2 ="hello str";
//将char * 转成 string (直接完成)
str1 = str2;
cout<<str1<<endl;//hello str
string str3="hello str3";
//不能直接将string 转换成 char * 必须借助string中的c_str方法完成
//char *str4 = str3;//err
char *str4 = const_cast<char *> (str3.c_str());
cout<<str4<<endl;//"hello str3"
}
int main(int argc, char *argv[])
{
test08();
return 0;
}
vector容器的迭代器 :随机访问迭代器
随机访问迭代器:迭代器+n 可以通过编译 就是随机访问迭代器
capacity:空间能容纳元素的最大个数。
size:空间中实际存放的元素个数
#include
#include
using namespace std;
void test01(){
vector<int> v;
int i = 0;
for(i = 0; i < 100; i++){
v.push_back(i);
}
cout << "v的容量capacity:" << v.capacity() << endl;
cout << "v的大小size:" << v.size() << endl;
}
int main(int argc, char *argv[])
{
test01();
return 0;
}
运行结果:
void test02(){
vector<int> v;
int *p = NULL;
int count = 0;
for(int i = 0; i < 1000; i++){
v.push_back(i);
if(p != &v[0]){
count++;
p = &v[0];
}
}
cout << "另寻地址的次数" << count << endl;
}
int main(int argc, char *argv[])
{
test02();
return 0;
}
运行结果:
只要空间满,就会另寻空间
void test03(){
vector<int> v;
int *p = NULL;
int count = 0;
for(int i = 0; i < 1000; i++){
if(p != &v[0]){
cout << "---------"<< count++ << "--------"<< endl;
p = &v[0];
}
v.push_back(i);
cout<<"cacapity = "<<v.capacity()<<", size = "<<v.size()<<endl;
}
}
int main(int argc, char *argv[])
{
test03();
return 0;
}
/*
* vector v; //采用模板实现类实现,默认构造函数
* vector(v.begin(), v.end());//将v[begin(), end())区间中的元素拷贝给本身。
* assign(beg, end);//将[beg, end)区间中的数据拷贝赋值给本身。
* assign(n, elem);//将n个elem拷贝赋值给本身。
* vector& operator=(const vector &vec);//重载等号操作符
*
*/
void printVectorInt(vector<int> &v){
for(vector<int>::iterator it=v.begin();it!=v.end();it++){
cout<<*it<<" ";
}
cout << endl;
}
void test05(){
//vector(n, elem);//构造函数将n个elem拷贝给本身
vector<int> v1 = vector<int>(10,5);
printVectorInt(v1);
//vector(v.begin(), v.end());//将v[begin(), end())区间中的元素拷贝给本身
vector<int> v2(v1.begin()+2,v1.end()-2);
printVectorInt(v2);
vector<int> v3(v1);
printVectorInt(v3);
//vector& operator=(const vector &vec);//重载等号操作符
v2 = v1;
printVectorInt(v2);
//assign(n, elem);//将n个elem拷贝赋值给本身
vector<int> v4;
v4.assign(5,100);
printVectorInt(v4);
//assign(beg, end);//将[beg, end)区间中的数据拷贝赋值给本身
vector<int> v5;
v4.assign(v4.begin(), v4.end());
printVectorInt(v4);
//swap(vec);// 将vec与本身的元素互换。
vector<int> v6(5,20);
vector<int> v7(10,40);
printVectorInt(v6);
printVectorInt(v7);
v6.swap(v7);
printVectorInt(v6);
printVectorInt(v7);
}
int main(int argc, char *argv[])
{
test05();
return 0;
}
运行结果:
/*
* size();//返回容器中元素的个数
* empty();//判断容器是否为空
* resize(int num);//重新指定容器的长度为num,若容器变长,则以默认值填充新位置。 如果容器变短,则末尾超出容器长度的元素被删除。
* resize(int num, elem);//重新指定容器的长度为num,若容器变长,则以elem值填充新 位置。如果容器变短,则末尾超出容器长>度的元素被删除。
* capacity();//容器的容量
* reserve(int len);//容器预留len个元素长度,预留位置不初始化,元素不可访问
*/
void test06(){
vector<int> v;
v.push_back(10);
v.push_back(20);
v.push_back(30);
v.push_back(40);
if(v.empty()){
cout<<"v容器为空"<<endl;
}else{
cout<<"容器非空"<<endl;
cout<<"size = "<<v.size()<<endl;
cout<<"capacity = "<<v.capacity()<<endl;
//容量 >= size
}
printVectorInt(v);//10 20 30 40
//resize(int num);//重新指定容器的长度为num
//多出的部分 自动补0
v.resize(8);
printVectorInt(v);//10 20 30 40 0 0 0 0
//resize(int num, elem);//重新指定容器的长度为num,
//若容器变长,则以elem值填充
v.resize(10,5);
printVectorInt(v);//10 20 30 40 0 0 0 0 5 5
v.resize(2);
printVectorInt(v);//10 20
//resize 作用的容器的大小 不会更改容器的容量
cout<<"size = "<<v.size()<<endl;
cout<<"capactiy = "<<v.capacity()<<endl;
}
int main(int argc, char *argv[])
{
test06();
return 0;
}
运行结果:
使用resize swap收缩容器的容量:
void test04()
{
vector<int> v;
for(int i=0;i<1000;i++)
{
v.push_back(i);
}
cout<<"size = "<<v.size()<<endl;//1000
cout<<"capactiy = "<<v.capacity()<<endl;//1024
//使用reszie将空间 置成10个元素(可以吗?)
v.resize(10);//不能修改容量 只能修改size
cout<<"size = "<<v.size()<<endl;//10
cout<<"capactiy = "<<v.capacity()<<endl;//1024
//使用swap收缩容器的容量
vector<int>(v).swap(v);
cout<<"size = "<<v.size()<<endl;//10
cout<<"capactiy = "<<v.capacity()<<endl;//10
}
运行结果:
void test09(){
//reserve(int len);//容器预留len个元素长度,预留位置不初始化,元素不可访问
vector<int> v;
//一次性 给够空间 叫空间预留
v.reserve(1000);//预留空间 1000个元素
int *p = NULL;
int count = 0;
for(int i = 0; i<1000; i++){
v.push_back(i);
if(p != &v[0]){
count++;
p = &v[0];
}
}
cout << "重新另寻空间次数" << count <<endl;
}
int main(int argc, char *argv[])
{
test09();
return 0;
}
运行结果:
如果没有:v.reserve(1000); 结果为11
如果有:v.reserve(1000); 结果为1
void printVectorInt(vector<int> &v){
for(vector<int>::iterator it=v.begin();it!=v.end();it++){
cout<<*it<<" ";
}
cout << endl;
}
void test10(){
vector<int> v;
v.push_back(10);
v.push_back(20);
v.push_back(30);
v.push_back(40);
printVectorInt(v);
cout<<v[2]<<endl;//30
cout<<v.at(2)<<endl;//30
//[] 越界 不抛出异常
//at 越界 抛出异常
cout<<"front = "<<v.front()<<endl;//10
cout<<"back = "<<v.back()<<endl;//40
}
int main(int argc, char *argv[])
{
test10();
return 0;
}
void printVectorInt(vector<int> &v){
for(vector<int>::iterator it=v.begin();it!=v.end();it++){
cout<<*it<<" ";
}
cout << endl;
}
void test11(){
vector<int> v;
v.push_back(10);
v.push_back(20);
v.push_back(30);
v.push_back(40);
printVectorInt(v);//10 20 30 40
//insert(const_iterator pos, int count,ele);
//迭代器指向位置pos插入count个元素ele.
v.insert(v.begin()+2,3,100);
printVectorInt(v);//10 20 100 100 100 30 40
//尾部删除:pop_back();//删除最后一个元素
v.pop_back();//将40删除了
printVectorInt(v);//10 20 100 100 100 30
//erase(const_iterator start, const_iterator end);
//删除迭代器从start到end之间的元素
v.erase(v.begin()+2,v.end()-1);
printVectorInt(v);//10 20 30
//erase(const_iterator pos);//删除迭代器指向的元素
v.erase(v.begin()+1);//删除20的位置
printVectorInt(v);//10 30
cout<<"size = "<<v.size()<<", capacity = "<<v.capacity()<<endl;
//clear();//删除容器中所有元素
v.clear();
printVectorInt(v);//啥也没有
cout<<"size = "<<v.size()<<", capacity = "<<v.capacity()<<endl;
}
int main(int argc, char *argv[])
{
test11();
return 0;
}
运行结果: