本文为B站黑马程序员 C++视频学习笔记,若侵权,则删,B站搜索黑马程序员可查看原视频。
STL诞生
1、长久以来,软件界一致希望建立一种可重复利用的东西
2、C++的面向对象和泛型编程思想,目的就是复用性的提升
3、大多情况下,数据结构和算法都未能有一套标准,导致被迫从事大量重复性工作
4、为了建立数据结构和算法的一套标准,诞生了STL
STL基本概念
1、STL(standard Template Library,标准模版库)
2、STL从广义上分为:容器(container)、算法(algorithm)、迭代器(iterator)
3、容器和算法之间通过迭代器进行无缝连接
4、STL几乎所有的代码都采用了模板类或者模板函数
STL六大组件
STL大体分为六大组件,分别为:容器、算法、迭代器、仿函数、适配器(配接器)、空间配置器
1、容器: 各种数据结构,如vector 、list、deque、set、map等,用来存放数据
2、算法: 各种常用算法,如sort、find、copy、for_each等
3、迭代器: 扮演了容器与算法之间的胶合剂
4、仿函数: 行为类似函数,可作为算法的某种策略
5、适配器: 一种用来修饰容器或者仿函数或迭代器接口的东西
6、空间配置器: 负责空间的配置与管理
STL中容器、算法、迭代器
容器: STL容器就是将运用最广泛的一些数据结构实现出来
常用的数据结构: 数组、链表、树、栈、队列、集合、映射表等
容器分为序列式容器和关联式容器两种:
序列式容器: 强调值的排序,序列式容器中的每个元素均有固定的位置
关联式容器: 二叉树结构,各元素之间没有严格的物理上的顺序关系
算法: 有限的步骤,解决逻辑或数学上的问题,叫做算法
算法分为:质变算法和非质变算法
质变算法: 是指运算过程中会更改区间内的元素的内容。例如拷贝、替换、删除等等
非质变算法: 是指运算过程中不会更改区间内的元素内容。例如查找、计数、遍历、寻找极值等
迭代器: 容器和算法之间的粘合剂
提供一种方法,使之能够依序寻访某个容器所含的各个元素,而又无需暴露该容器的内部表示方式
每个容器都有自己专属的迭代器
迭代器使用非常类似于指针,初学阶段可以先理解为迭代器为指针
迭代器种类:
常见的容器中迭代器种类为双向迭代器和随机访问迭代器
vector基本概念
功能: vector数据结构和数组非常相似,也称为单端数组
与普通数组的区别:数组是静态空间,vector可以动态扩展
动态扩展: 并不是在原空间之后续接新空间,而是找更大的内存空间,然后将原数据拷贝到新空间,释放原空间
vector容器的迭代器是支持随机访问的迭代器。
vector 构造函数
函数原型:
vector<T> v; //采用模板实现类实现,默认构造函数
vector(v.begin(),v.end());//将v[begin(),end()]区间中的元素拷贝给本身
vector(n,elem); //构造函数将n个elem拷贝给本身
vector(const vector &vec); //拷贝构造函数
示例:
#include <stdio.h>
#include<iostream>
#include<vector>
using namespace std;
void printVector(vector<int>&v)
{
for(vector<int>::iterator it = v.begin(); it != v.end(); it ++)
{
cout << (*it) <<" ";
}
cout<<endl;
}
//vector 容器构造
void test01()
{
//vector<T> v; 采用模板实现类实现,默认构造函数,无参构造
vector<int>v1;
for(int i = 0; i < 10; i ++)
{
v1.push_back(i);
}
printVector(v1);
//通过区间方式进行构造 vector(v.begin(),v.end()); 将v[begin(),end()]区间中的元素拷贝给本身
vector<int>v2(v1.begin(),v1.end());
printVector(v2);
//vector(n,elem); 构造函数将n个elem拷贝给本身
vector<int>v3(10,100);//10个100
printVector(v3);
//vector(const vector &vec); 拷贝构造函数
vector<int>v4(v3);
printVector(v4);
}
int main()
{
test01();
return 0;
}
vector赋值操作
函数原型:
vector& operator=(const vector &vec); //重载等号 运算符
assign(beg,end);//将(beg,end]区间中的数据 拷贝赋值给本身
assign(n,elem); //将n个elem拷贝赋值给本身
示例:
#include <stdio.h>
#include<iostream>
#include<vector>
using namespace std;
void printVector(vector<int>&v)
{
for(vector<int>::iterator it = v.begin(); it != v.end(); it ++)
{
cout << (*it) <<" ";
}
cout<<endl;
}
//vector赋值
void test01()
{
//vector<T> v; 采用模板实现类实现,默认构造函数,无参构造
vector<int>v1;
for(int i = 0; i < 10; i ++)
{
v1.push_back(i);
}
printVector(v1);
//赋值 operator=
vector<int>v2;
v2 = v1;
printVector(v2);
//assign
vector<int>v3;
v3.assign(v1.begin(),v1.end());
printVector(v2);
//assign(n,elem)
vector<int>v4;
v4.assign(10,100);
printVector(v4);
}
int main()
{
test01();
return 0;
}
vector容量和大小
函数原型:
empty(); //判断容器是否为空
capacity();//容器的容量
size(); //返回容器中元素的个数
resize(int num);//重新指定容器的长度为num,若容器变长,则以默认值填充新位置
//如果容器变短,则末尾超出容器长度的元素被删除
resize(int num,elem); //重新指定容器的长度为num,若容器变长,则以elem值填充新位置
//如果容器变短,则末尾超出容器长度的元素被删除
示例:
#include <stdio.h>
#include<iostream>
#include<vector>
using namespace std;
void printVector(vector<int>&v)
{
for(vector<int>::iterator it = v.begin(); it != v.end(); it ++)
{
cout << (*it) <<" ";
}
cout<<endl;
}
//vector容器的容量和大小操作
void test01()
{
vector<int>v1;
for(int i = 0; i < 10; i ++)
{
v1.push_back(i);
}
printVector(v1);
if(v1.empty()) //为真,代表容器为空
{
cout<<"v1为空"<<endl;
}
else
{
cout<<"v1不为空"<<endl;
cout<<"v1的容量为:"<<v1.capacity()<<endl;
cout<<"v1的大小为:"<<v1.size()<<endl;
}
v1.resize(15/*,100*/); //可指定 默认填充值
printVector(v1); //output : 0 1 2 3 4 5 6 7 8 9 0 0 0 0
//如果重新指定比原来长,默认用0填充新位置
v1.resize(5);
printVector(v1); //如果重新指定的比原来短了,超出部分会删除掉
}
int main()
{
test01();
return 0;
}
vector插入和删除
函数原型:
push_back(ele); //尾部插入元素ele
pop_back(); //删除最后一个元素
insert(const_iterator pos,ele); //迭代器指向位置pos插入元素ele
insert(const_iterator pos,int count,ele); //迭代器指向位置pos插入count个元素ele
erase(const_iterator pos); //删除迭代器指向的元素
erase(const_iterator start,const_iterator end); //删除迭代器从start到end之间的元素
clear(); //删除容器中所有的元素
示例:
#include <stdio.h>
#include<iostream>
#include<vector>
using namespace std;
void printVector(vector<int>&v)
{
for(vector<int>::iterator it = v.begin(); it != v.end(); it ++)
{
cout << (*it) <<" ";
}
cout<<endl;
}
//vector容器的容量和大小操作
void test01()
{
vector<int>v1;
//尾插
v1.push_back(10);
v1.push_back(20);
v1.push_back(30);
v1.push_back(40);
v1.push_back(50);
//遍历
printVector(v1); //output : 10 20 30 40 50
//尾删
v1.pop_back();
printVector(v1); //output : 10 20 30 40
//插入 第一个参数是迭代器
v1.insert(v1.begin(),100);
printVector(v1); //output : 100 10 20 30 40
//insert(const_iterator pos,int count,ele); //迭代器指向位置pos插入count个元素ele
v1.insert(v1.begin(),2,1000);
printVector(v1); //output : 1000 1000 100 10 20 30 40
//删除
v1.erase(v1.begin());
printVector(v1); //output : 1000 100 10 20 30 40
//清空
v1.erase(v1.begin(),v1.end());
//v1.clear();
printVector(v1); //output :
}
int main()
{
test01();
return 0;
}
vector数据存取
函数原型:
at(int idx); //返回索引idx所指的数据
operator[]; //返回索引idx所指的数据
front(); //返回容器中第一个数据元素
back(); //返回容器中最后一个数据元素
示例:
#include <stdio.h>
#include <iostream>
using namespace std;
#include <vector>
//vector容器 数据存取
void test01()
{
vector<int>v1;
for(int i = 0 ;i < 10; i++)
{
v1.push_back(i);
}
//利用[]方式访问数组中元素
for(int i = 0 ;i < 10; i++ )
{
cout<<v1[i]<<" ";
}
cout<<endl;
//利用at()方式访问数组中元素
for(int i = 0 ;i < 10; i++ )
{
cout<<v1.at(i)<<" ";
}
cout<<endl;
//获取第一个元素
cout<<"第一个元素为:"<<v1.front() <<endl;
//获取最后一个元素
cout<<"第一个元素为:"<<v1.back() <<endl;
}
int main()
{
test01();
return 0;
}
vector存放内置数据类型
容器:vector
算法:for_each
迭代器:vector< int >::iterator
#include <iostream>
using namespace std;
#include<vector>
#include<algorithm> //标准算法头文件
void myPrint(int val)
{
cout<<val<<endl;
}
//vector容器存放内置数据类型
void test01()
{
//创建了一个vector容器,数组
vector<int> v;
//向容器中插入数据
v.push_back(10);
v.push_back(20);
v.push_back(30);
v.push_back(40);
//通过迭代器访问容器中的数据
vector<int>::iterator itBegin = v.begin(); //起始迭代器 指向容器中第一个元素
vector<int>::iterator itEnd = v.end(); //结束迭代器 指向容器中最后一个元素的下一个位置
//第一种遍历方式
/* while(itBegin!=itEnd)
{
cout<<*itBegin<<endl;
itBegin++;
}
*/
/*
//第二种遍历方式
for(vector<int>::iterator it = v.begin(); it != v.end();it++)
{
cout << *it <<endl;
}
*/
//第三种遍历方式 利用STL提供遍历算法 需包含#include<algorithm>
for_each(v.begin(),v.end(),myPrint); //利用回调函数进行遍历
}
int main()
{
test01();
return 0;
}
vector存放自定义数据类型
#include <iostream>
using namespace std;
#include<vector>
#include<string>
#include<algorithm> //标准算法头文件
class Person
{
public:
string m_Name;
int m_Age;
public:
Person(string name, int age)
{
this -> m_Name = name;
this -> m_Age = age;
}
};
//vector容器存放Person 对象
void test01()
{
//创建了一个vector容器,数组
vector<Person> v;
Person p1("aaa",10);
Person p2("bbb",20);
Person p3("ccc",30);
Person p4("ddd",40);
Person p5("eee",50);
//向容器中插入数据
v.push_back(p1);
v.push_back(p2);
v.push_back(p3);
v.push_back(p4);
v.push_back(p5);
//遍历 (*it)的数据类型和<Person>(尖括号中数据类型相同)
for(vector<Person>::iterator it = v.begin(); it != v.end();it++)
{
cout << "姓名:"<<(*it).m_Name <<"年龄:" <<(*it).m_Age<<endl;
cout << "姓名:"<<it->m_Name <<"年龄:" <<it->m_Age<<endl;
}
}
//存放自定义数据类型 指针
void test02()
{
vector<Person*>v;
Person p1("aaa",10);
Person p2("bbb",20);
Person p3("ccc",30);
Person p4("ddd",40);
Person p5("eee",50);
//向容器中插入Person对象的地址
v.push_back(&p1);
v.push_back(&p2);
v.push_back(&p3);
v.push_back(&p4);
v.push_back(&p5);
//遍历
for(vector<Person*>::iterator it = v.begin(); it != v.end();it++)
{
cout << "姓名:"<<(*it)->m_Name <<"年龄:" <<(*it)->m_Age<<endl;
}
}
int main()
{
test02();
return 0;
}
vector容器嵌套容器
#include <iostream>
using namespace std;
#include<vector>
#include<string>
#include<algorithm> //标准算法头文件
//容器嵌套容器
void test01()
{
//创建了一个vector容器,数组
vector<vector<int>> v;
//创建小容器
vector<int>v1;
vector<int>v2;
vector<int>v3;
vector<int>v4;
//向小容器中添加数据
for(int i = 0; i < 4; i++)
{
v1.push_back(i + 1);
v2.push_back(i + 2);
v3.push_back(i + 3);
v4.push_back(i + 4);
}
//将小容器插入到大容器中
v.push_back(v1);
v.push_back(v2);
v.push_back(v3);
v.push_back(v4);
//通过大容器把所有数据遍历
for(vector<vector<int>>::iterator it = v.begin(); it != v.end();it++)
{
for(vector<int>::iterator vit = (*it).begin(); vit != (*it).end();vit++)
{
cout<<(*vit)<<" ";
}
cout<<endl;
}
}
int main()
{
test01();
return 0;
}
vector互换容器
函数原型:
swap(vec); //将vec与本身元素互换
示例:
#include <stdio.h>
#include <iostream>
using namespace std;
#include <vector>
//vector容器互换
void printVector(vector<int>&v)
{
for(vector<int>::iterator it = v.begin(); it != v.end(); it ++)
{
cout << (*it) <<" ";
}
cout<<endl;
}
//基本使用
void test01()
{
vector<int>v1;
for(int i = 0 ;i < 10; i++)
{
v1.push_back(i);
}
cout<<"交换前:"<<endl;
printVector(v1);
vector<int>v2;
for(int i = 10 ;i > 0; i--)
{
v2.push_back(i);
}
printVector(v2);
cout<<"交换后:"<<endl;
v1.swap(v2);
printVector(v1);
printVector(v2);
}
//实际用途
//巧用swap可以收缩内存空间
void test02()
{
vector<int>v;
for(int i = 0; i<100000; i++)
{
v.push_back(i);
}
cout<<"v的容量是:"<<v.capacity()<<endl; //output:138255
cout<<"v的大小为:"<<v.size()<<endl; //output:100000
v.resize(3);//重新指定大小
cout<<"v的容量是:"<<v.capacity()<<endl; //output:138255
cout<<"v的大小为:"<<v.size()<<endl; //output:3
//巧用swap收缩内存
vector<int>v.swap(v); //vector<int>v ---- 匿名对象x(拷贝构造) swap()互换匿名对象和v
//匿名对象执行完系统自动回收 回收x 进而收缩内存
cout<<"v的容量是:"<<v.capacity()<<endl; //output:3
cout<<"v的大小为:"<<v.size()<<endl; //output:3
}
int main()
{
test01();
return 0;
}
总结 :
swap可以使两个内存互换,可以达到实用的收缩内存的功效
vector预留空间
函数原型:
reserve(int len); //容器预留len个元素长度,预留位置不初始化,元素不可访问
示例:
#include <stdio.h>
#include <iostream>
using namespace std;
#include <vector>
//vector容器 预留空间
void printVector(vector<int>&v)
{
for(vector<int>::iterator it = v.begin(); it != v.end(); it ++)
{
cout << (*it) <<" ";
}
cout<<endl;
}
//基本使用
void test01()
{
vector<int>v;
v.reserve(100000); //预留空间
int num = 0; //统计开辟次数
int *p = NULL;
for(int i = 0 ;i < 100000; i++)
{
v.push_back(i);
if(p != &v[0])
{
p = &v[0];
num++;
}
}
cout <<"num = "<< num <<endl; // output:30 开辟了30次 可通过预留空间进行次数减少
}
int main()
{
test01();
return 0;
}
总结: 如果数据量较大,可以一开始i利用reserve预留空间