C++从入门到精通 第十四章(STL容器)【上】

 写在前面:

  1. 本系列专栏主要介绍C++的相关知识,思路以下面的参考链接教程为主,大部分笔记也出自该教程,笔者的原创部分主要在示例代码的注释部分。
  2. 除了参考下面的链接教程以外,笔者还参考了其它的一些C++教材(比如计算机二级教材和C语言教材),笔者认为重要的部分大多都会用粗体标注(未被标注出的部分可能全是重点,可根据相关部分的示例代码量和注释量判断,或者根据实际经验判断)。
  3. 如有错漏欢迎指出。

参考教程:黑马程序员匠心之作|C++教程从0到1入门编程,学习编程不再难_哔哩哔哩_bilibili

一、string容器

1、string容器概述

(1)string的本质:string是C++风格的字符串,而string本质上是一个类。

(2)string和char *的区别:char *是一个指针,而string是一个类,类内部封装了char*,管理这个字符串,是一个char*型的容器。

(3)string 类内部封装了很多成员方法,例如查找find、拷贝copy、删除delete、替换replace、插入insert。

(4)string管理char*所分配的内存,不用担心复制越界和取值越界等,由类内部进行负责。

2、string类的构造函数

string();                //创建一个空的字符串(默认构造)

string(const string& str);   //使用一个string对象初始化另一个string对象(拷贝构造)

string(int n, char c);       //使用n个字符c初始化string对象

#include<iostream>
#include<string>
using namespace std;

void test01()  
{
	string s1;      //默认构造

	const char * str = "Hello world";    //使用字符串初始化
	string s2(str);                      //这个也可视作拷贝构造
	cout << "s2 : " << s2 << endl;

	string s3(s2);   //拷贝构造
	cout << "s2 : " << s2 << endl;

	string s4(10, 'a');   //使用n个字符初始化
	cout << "s4 : " << s4 << endl;
}

int main() {

	test01();

	system("pause");

	return 0;
}

3、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组成字符串赋给当前的字符串

#include<iostream>
#include<string>
using namespace std;
#include<vector>

void test01()   
{
	string str1;
	str1 = "Hello world";
	cout << "srt1 = " << str1 << endl;

	string str2;
	str2 = str1;
	cout << "srt2 = " << str2 << endl;

	string str3;
	str3 = 'a';
	cout << "srt3 = " << str3 << endl;

	string str4;
	str4.assign("Hello C++");
	cout << "srt4 = " << str4 << endl;

	string str5;
	str5.assign("Hello C++", 5);
	cout << "srt5 = " << str5 << endl;

	string str6;
	str6.assign(str5);
	cout << "srt6 = " << str6 << endl;

	string str7;
	str7.assign(10, 'w');
	cout << "srt7 = " << str7 << endl;
}

int main() {

	test01();

	system("pause");

	return 0;
}

4、string字符串拼接函数

string& operator+=(const char* str);  //重载+=操作符

string& operator+=(const char c);    //重载+=操作符

string& operator+=(const string& str); //重载+=操作符

string& append(const char *s);      //把字符串s连接到当前字符串结尾

string& append(const char *s, int n);  //把字符串s的前n个字符连接到当前字符串结尾

string& append(const string &s);     //同operator+=(const string& str)

string& append(const string &s, int pos, int n);//字符串s中从pos开始的n个字符连接到字符串结尾

#include<iostream>
#include<string>
using namespace std;

void test01()   
{
	string str1;
	str1 = "我";
	str1 += "爱玩游戏";
	cout << "str1 = " << str1 << endl;
	str1 += ':';
	cout << "str1 = " << str1 << endl;
	string str2 = " LOL DNF";
	str1 += str2;
	cout << "str1 = " << str1 << endl;

	string str3 = "I";
	str3.append(" love ");
	cout << "str3 = " << str3 << endl;
	str3.append("Python abcde", 6);
	cout << "str3 = " << str3 << endl;
	str3.append(str2);
	cout << "str3 = " << str3 << endl;
	str3.append(str2, 0, 4);     //参数2:从哪个位置开始截取
	cout << "str3 = " << str3 << endl;
	str3.append(str2, 4, 4);     //参数3:截取字符个数
	cout << "str3 = " << str3 << endl;
}

int main() {

	test01();

	system("pause");

	return 0;
}

5、string类的查找和替换函数

查找指的是查找指定字符串是否存在,替换指的是在指定的位置替换字符串。

int find(const string& str, int pos = 0) const; //查找str第一次出现位置,从pos开始查找

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最后一次出现位置,从pos开始查找

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

#include<iostream>
#include<string>
using namespace std;

void test01()     //查找
{
	string str1 = "abcdefgcd";        //该字符串有两个“cd”
	int pos = str1.find("cd");      //find从左往右查
	if (pos == -1)
	{
		cout << "未找到字符串" << endl;
	}
	else
	{
		cout << "pos = " << pos << endl;
	}
	pos = str1.find("df");
	if (pos == -1)
	{
		cout << "未找到字符串" << endl;
	}
	else
	{
		cout << "pos = " << pos << endl;
	}
	
	pos = str1.rfind("cd");       //rfind从右往左查(位置编号还是从左往右算)
	cout << "pos = " << pos << endl;
	
}

void test02()     //替换
{
	string str1 = "abcdefg";
	str1.replace(2, 3, "kkkk");    //从2号位置起的3个字符替换为“kkkk”
	cout << "str1 = " << str1 << endl;
}

int main() {

	test01();
	test02();

	system("pause");

	return 0;
}

6、string字符串比较函数

比较方式:字符串比较是从第一个字符开始,按字符的ASCII码进行对比,只要出现同位置字符ASCII码不同的情况,那么两个字符串就不相等,认为ASCII码较大的一个字符串更大。

①比较结果为“=”,返回0。

②比较结果为“>”,返回1。

③比较结果为“<”,返回-1。

int compare(const string &s) const;  //与字符串s比较

int compare(const char *s) const;   //与字符串s比较

#include<iostream>
#include<string>
using namespace std;

string compare(string &str1,string &str2)
{
	if (str1.compare(str2) == 0)
	{
		return " 等于= ";
	}
	else if (str1.compare(str2) == 1)
	{
		return " 大于> ";
	}
	else if(str1.compare(str2) == -1)
	{
		return " 小于< ";
	}
}

void test01()     
{
	string str1 = "hello";
	string str2 = "hello";
	string str3 = "xello";
	string str4 = "heklo";
	cout << "str1" << compare(str1, str2) << "str2" << endl;
	cout << "str1" << compare(str1, str3) << "str3" << endl;
	cout << "str1" << compare(str1, str4) << "str4" << endl;
}

int main() {

	test01();

	system("pause");

	return 0;
}

7、string字符串的字符存取

string字符串中单个字符存取方式有两种:

char& operator[](int n);  //通过[]方式取字符

char& at(int n);        //通过at方法获取字符

#include<iostream>
#include<string>
using namespace std;

void test01()     
{
	string str = "Hello world";
	//通过[]访问单个字符
	for (int i = 0; i < str.size(); i++)
	{
		cout << str[i] << " ";
	}
	cout << endl;
	str[0] = 'h';
	//通过at方式访问单个字符
	for (int i = 0; i < str.size(); i++)
	{
		cout << str.at(i) << " ";
	}
	cout << endl;
	str.at(0) = 'H';
	for (int i = 0; i < str.size(); i++)
	{
		cout << str.at(i) << " ";
	}
	cout << endl;
}

int main() {

	test01();

	system("pause");

	return 0;
}

8、string字符串的字符插入和字符删除

对string字符串进行插入和删除操作时,要将string字符串视为字符数组,起始的下标都是从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个字符

#include<iostream>
#include<string>
using namespace std;

void test01()     
{
	string str = "Hello";
	str.insert(1, "k2k");   //插入
	cout << "str = " << str << endl;
	str.erase(1, 3);        //删除
	cout << "str = " << str << endl;
}

int main() {

	test01();

	system("pause");

	return 0;
}

9、从字符串string获取子串

string substr(int pos = 0, int n = npos) const; //返回由pos开始的n个字符组成的字符串

#include<iostream>
#include<string>
using namespace std;

void test01()     
{
	string str = "abcdefg";
	string subStr = str.substr(1, 3);
	cout << "subStr = " << subStr << endl;
}
void test02()
{
	string email = "zhangsan@sina.com";
	int pos = email.find("@");
	string usrName = email.substr(0, pos);
	cout << "usrName = " << usrName << endl;
}

int main() {

	test01();
	test02();

	system("pause");

	return 0;
}

二、vector容器

1、vector的基本概念

(1)vector数据结构和数组非常相似,也称为单端数组。

(2)vector与普通数组区别:数组是静态空间,而vector可以动态扩展,而动态扩展并不是在原空间之后续接新空间,而是找更大的内存空间,然后将原数据拷贝新空间,释放原空间。

(3)vector容器的迭代器是支持随机访问的迭代器。

2、vector的构造函数

vector<T> v;              //采用模板实现类实现,默认构造函数

vector(v.begin(), v.end());    //将v[begin(), end())区间中的元素拷贝给本身

vector(n, elem);           //构造函数将n个elem拷贝给本身

vector(const vector &vec);  //拷贝构造函数

#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;
}

void test01()     
{
	vector<int> v1;                          //默认构造(无参构造)
	for (int i = 0; i < 10; i++)
	{
		v1.push_back(i);
	}
	printVector(v1);
	vector<int> v2(v1.begin(), v1.end());    //通过区间方式进行构造
	printVector(v2);
	vector<int> v3(10, 199);                 //n个elem方式构造
	printVector(v3);
	vector<int> v4(v3);                      //拷贝构造
	printVector(v4);
}

int main() {

	test01();

	system("pause");

	return 0;
}

3、vector的赋值操作

vector& operator=(const vector &vec); //重载等号操作符

assign(beg, end);                  //将[beg, end)区间中的数据拷贝赋值给本身

assign(n, elem);                   //将n个elem拷贝赋值给本身

#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;
}

void test01()     
{
	vector<int> v1;
	for (int i = 0; i < 10; i++)
	{
		v1.push_back(i);
	}
	printVector(v1);
	vector<int>v2;
	v2 = v1;          //赋值 operator=
	printVector(v2);
	vector<int>v3;
	v3.assign(v1.begin(), v1.end());   //赋值 assign
	printVector(v3);
	vector<int>v4;
	v4.assign(10, 51);                 //赋值 n个elem方式
	printVector(v4);
}

int main() {

	test01();

	system("pause");

	return 0;
}

4、vector的容量和大小

empty();       //判断容器是否为空

capacity();     //容器的容量

size();         //返回容器中元素的个数

resize(int num); //重新指定容器的长度为num,若容器变长,则以默认值填充新位置

​ //如果容器变短,则末尾超出容器长度的元素被删除

resize(int num, elem); //重新指定容器的长度为num,若容器变长,则以elem值填充新位置

​ //如果容器变短,则末尾超出容器长度的元素被删除

#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;
}

void test01()     
{
	vector<int> v1,v2;
	if (v1.empty())
	{
		cout << "容器v1为空" << endl;
	}
	cout << endl;
	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);
	printVector(v1);    //如果重新指定的长度比原来长了,默认用0填充新位置
	cout << "v1的容量为" << v1.capacity() << endl;
	cout << "v1的元素个数为" << v1.size() << endl;
	v2 = v1;
	v2.resize(8);
	printVector(v2);    //如果重新指定的长度比原来短,则超出长度的元素会被删除
	v2.resize(12, 10);
	printVector(v2);    //重新指定容器的长度,若容器变长,则以elem值填充新位置
}

int main() {

	test01();

	system("pause");

	return 0;
}

5、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<iostream>
#include<vector>
using namespace std;

void printVector(vector<int> &v)
{
	int j = 0;
	for (vector<int>::iterator it = v.begin(); it != v.end(); it++)
	{
		cout << *it << "  ";
		j++;
	}
	if (j == 0)
	{
		cout << "什么也没有" << endl;
	}
	cout << endl;
}

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);
	v1.pop_back();    //尾删
	printVector(v1);
	v1.insert(v1.begin(), 100);   //插入  第一个参数是迭代器
	printVector(v1);
	v1.insert(v1.begin(), 2 ,100);
	printVector(v1);
	v1.erase(v1.begin());         //删除 参数是迭代器
	printVector(v1);
	v1.erase(v1.begin(), v1.end());   //v1.clear();具有同等作用
	printVector(v1);
}

int main() {

	test01();

	system("pause");

	return 0;
}

6、vector的数据存取

at(int idx);   //返回索引idx所指的数据

operator[];  //返回索引idx所指的数据

front();     //返回容器中第一个数据元素

back();     //返回容器中最后一个数据元素

#include<iostream>
#include<vector>
using namespace std;

void test01()     
{
	vector<int> v1;
	for (int i = 0; i < 10; i++)
	{
		v1.push_back(i);
	}
	//利用[]方式访问数组元素
	for (int i = 0; i < v1.size(); i++)
	{
		cout << v1[i] << " ";
	}
	cout << endl;
	//利用at方式访问元素
	for (int i = 0; i < v1.size(); i++)
	{
		cout << v1.at(i) << " ";
	}
	cout << endl;
	cout << "第一个元素为: " << v1.front() << endl;
	cout << "最后一个元素为: " << v1.back() << endl;
}

int main() {

	test01();

	system("pause");

	return 0;
}

7、vector的互换容器操作

swap(vec);   //将vec中的元素与调用该函数的对象本身的元素互换

#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;
}

void test01()     
{
	cout << "交换前:" << endl;
	vector<int> v1;
	for (int i = 0; i < 10; i++)
	{
		v1.push_back(i * 10);
	}
	printVector(v1);
	vector<int> v2;
	for (int i = 9; i > -1; i--)
	{
		v2.push_back(i * 10);
	}
	printVector(v2);
	cout << "交换后:" << endl;
	v1.swap(v2);
	printVector(v1);
	printVector(v2);
}

void test02()      //巧用swap可以收缩内存空间
{
	vector<int>v;
	for (int i = 0; i < 100000; i++)
	{
		v.push_back(i);
	}
	cout << "v的容量为:" << v.capacity() << endl;
	cout << "v的大小为:" << v.size() << endl;
	v.resize(3);   //重新指定大小
	cout << "v的容量为:" << v.capacity() << endl;   //容量不会变小
	cout << "v的大小为:" << v.size() << endl;
	vector<int>(v).swap(v);            //vector<int>(v)--匿名对象,按照v的元素给它初始化,本行过后匿名对象被销毁
	cout << "v的容量为:" << v.capacity() << endl;   //容量变小
	cout << "v的大小为:" << v.size() << endl;
}

int main() {

	test01();
	test02();

	system("pause");

	return 0;
}

8、vector允许预留空间

预留空间也就是先在内存划分一片属于vector对象的空间,这部分空间可以不立即使用,这样可以减少vector在动态扩展容量时的扩展次数。

reserve(int len);   //容器预留len个元素长度,预留位置不初始化,元素不可访问

#include<iostream>
#include<vector>
using namespace std;

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;
}

int main() {

	test01();

	system("pause");

	return 0;
}

三、deque容器

1、deque容器的基本概念

(1)deque容器是一个双端数组,可以对头端进行插入删除操作。

(2)deque与vector的区别:

①vector在头部插入或删除元素的效率低,数据量越大,效率越低。

②deque在头部插入元素或删除元素的速度比vector快。

③vector访问元素时的速度会比deque快,这和两者内部实现有关。

(3)deque内部的工作原理:

①deque内部有个中控器维护每段缓冲区中的内容,缓冲区中存放真实数据。

②中控器维护的是每个缓冲区的地址,使得使用deque时像是使用一片连续的内存空间。

(4)deque容器的迭代器也是支持随机访问的。

2、deque的构造函数

deque<T> deqT;           //默认构造形式

deque(beg, end);          //构造函数将[beg, end)区间中的元素拷贝给本身

deque(n, elem);           //构造函数将n个elem拷贝给本身

deque(const deque &deq);  //拷贝构造函数

#include<iostream>
#include<deque>
using namespace std;

void printDeque(const deque<int> &d)
{
	for (deque<int>::const_iterator it = d.begin(); it != d.end(); it++)
	{
		//*it = 100;   容器中的数据不可以修改了
		cout << *it << "  ";
	}
	cout << endl;
}

void test01()     
{
	deque<int>d1;
	for (int i = 0; i < 10; i++)
	{
		d1.push_back(i);
	}
	printDeque(d1);
	deque<int>d2(d1.begin(), d1.end());
	printDeque(d2);
	deque<int>d3(10, 100);
	printDeque(d3);
	deque<int>d4(d3);
	printDeque(d4);
}

int main() {

	test01();

	system("pause");

	return 0;
}

3、deque的赋值操作

deque& operator=(const deque &deq);  //重载等号操作符

assign(beg, end);                    //将[beg, end)区间中的数据拷贝赋值给本身

assign(n, elem);                     //将n个elem拷贝赋值给本身

#include<iostream>
#include<deque>
using namespace std;

void printDeque(const deque<int> &d)
{
	for (deque<int>::const_iterator it = d.begin(); it != d.end(); it++)
	{
		cout << *it << "  ";
	}
	cout << endl;
}

void test01()     
{
	deque<int>d1;
	for (int i = 0; i < 10; i++)
	{
		d1.push_back(i);
	}
	printDeque(d1);

	//operator=赋值
	deque<int>d2;
	d2 = d1;
	printDeque(d2);

	//assign赋值
	deque<int>d3;
	d3.assign(d1.begin(), d1.end());
	printDeque(d3);

	//n个elem方式赋值
	deque<int>d4(10, 100);
	printDeque(d4);
}

int main() {

	test01();

	system("pause");

	return 0;
}

4、deque的大小

deque.empty();    //判断容器是否为空

deque.size();      //返回容器中元素的个数

deque.resize(num); //重新指定容器的长度为num,若容器变长,则以默认值填充新位置

​ //如果容器变短,则末尾超出容器长度的元素被删除。

deque.resize(num, elem); //重新指定容器的长度为num,若容器变长,则以elem值填充新位置

​ //如果容器变短,则末尾超出容器长度的元素被删除

#include<iostream>
#include<deque>
using namespace std;

void printDeque(const deque<int> &d)
{
	for (deque<int>::const_iterator it = d.begin(); it != d.end(); it++)
	{
		cout << *it << "  ";
	}
	cout << endl;
}

void test01()     
{
	deque<int>d1;
	for (int i = 0; i < 10; i++)
	{
		d1.push_back(i);
	}
	printDeque(d1);
	if (d1.empty())
	{
		cout << "容器d1为空" << endl;
	}
	cout << "d1中的元素个数为:" << d1.size() << endl;   //deque没有“容量”的概念
	d1.resize(15);
	printDeque(d1);
	d1.resize(8);
	printDeque(d1);
	d1.resize(15, 1);
	printDeque(d1);
}

int main() {

	test01();

	system("pause");

	return 0;
}

5、deque的插入和删除

//两端的插入操作:

push_back(elem);   //在容器尾部添加一个数据

push_front(elem);   //在容器头部插入一个数据

pop_back();        //删除容器最后一个数据

pop_front();        //删除容器第一个数据

//指定位置操作:

insert(pos,elem);      //在pos位置插入一个elem元素的拷贝,返回新数据的位置

insert(pos,n,elem);    //在pos位置插入n个elem数据,无返回值

insert(pos,beg,end);   //在pos位置插入[beg,end)区间的数据,无返回值

clear();              //清空容器的所有数据

erase(beg,end);       //删除[beg,end)区间的数据,返回下一个数据的位置

erase(pos);          //删除pos位置的数据,返回下一个数据的位置

#include<iostream>
#include<deque>
using namespace std;

void printDeque(const deque<int> &d)
{
	for (deque<int>::const_iterator it = d.begin(); it != d.end(); it++)
	{
		cout << *it << "  ";
	}
	cout << endl;
}

void test01()     
{
	deque<int>d1;
	for (int i = 0; i < 10; i++)
	{
		d1.push_back(i);      //尾插
		d1.push_front(i * 2); //头插
	}
	printDeque(d1);
	d1.pop_back();
	printDeque(d1);
	d1.pop_front();
	printDeque(d1);
}
void test02()
{
	deque<int>d1;
	for (int i = 0; i < 10; i++)
	{
		d1.push_back(i);
	}
	d1.insert(d1.begin(), 30);
	printDeque(d1);
	d1.insert(d1.begin(), 2, 40);
	printDeque(d1);
	deque<int>d2;
	for (int i = 0; i < 10; i++)
	{
		d2.push_front(i);
	}
	d1.insert(d1.begin()+1, d2.begin(), d2.end());
	printDeque(d1);
}
void test03()
{
	deque<int>d1;
	for (int i = 0; i < 10; i++)
	{
		d1.push_back(i);
	}
	d1.erase(++d1.begin());
	printDeque(d1);
	d1.erase(d1.begin(), d1.end());   //d1.clear();具有同等作用
	printDeque(d1);
}

int main() {

	test01();
	cout << endl;
	test02();
	cout << endl;
	test03();

	system("pause");

	return 0;
}

6、deque的数据存取

at(int idx);  //返回索引idx所指的数据

operator[];  //返回索引idx所指的数据

front();     //返回容器中第一个数据元素

back();     //返回容器中最后一个数据元素

#include<iostream>
#include<deque>
using namespace std;

void test01()     
{
	deque<int>d1;
	for (int i = 0; i < 10; i++)
	{
		d1.push_back(i);      //尾插
		d1.push_front(i * 2); //头插
	}
	//通过[]方式访问元素
	for (int i = 0; i < d1.size(); i++)
	{
		cout << d1[i] << " ";
	}
	cout << endl;
	//通过at方式访问元素
	for (int i = 0; i < d1.size(); i++)
	{
		cout << d1.at(i) << " ";
	}
	cout << endl;
	cout << "第一个元素为:" << d1.front() << endl;
	cout << "最后一个元素为:" << d1.back() << endl;
}

int main() {

	test01();

	system("pause");

	return 0;
}

7、deque的数据排序算法

sort(iterator beg, iterator end)    //对beg和end区间内元素进行排序

#include<iostream>
#include<deque>
#include<algorithm>
using namespace std;

void printDeque(const deque<int> &d)
{
	for (deque<int>::const_iterator it = d.begin(); it != d.end(); it++)
	{
		cout << *it << "  ";
	}
	cout << endl;
}

void test01()     
{
	deque<int>d1;
	for (int i = 0; i < 10; i++)
	{
		d1.push_back(i);      //尾插
		d1.push_front(i * 2); //头插
	}
	printDeque(d1);
	sort(d1.begin(),d1.end());   //升序排序
	//对于支持随机访问的迭代器的容器,都可以用sort算法直接对其进行排序
	//vector容器也可以利用sort进行排序
	printDeque(d1);
}

int main() {

	test01();

	system("pause");

	return 0;
}

8、案例

(1)案例描述:有5名选手ABCDE,10个评委分别对每一名选手打分,去除最高分,去除评委中最低分,取平均分。

(2)实现步骤:

①创建五名选手,放到vector中。

②遍历vector容器,取出来每一个选手,执行for循环,可以把10个评分打分存到deque容器中。

③sort算法对deque容器中分数排序,去除最高和最低分。

④deque容器遍历一遍,累加总分。

⑤获取平均分。

(3)代码:

#include<iostream>
#include<deque>
#include<vector>
#include<algorithm>
#include<string>
#include<time.h>
using namespace std;

/* 案例描述:有5名选手ABCDE,10个评委分别对每一名选手打分,去除最高分,去除评委中最低分,取平均分
实现步骤:
创建五名选手,放到vector中
遍历vector容器,取出来每一个选手,执行for循环,可以把10个评分打分存到deque容器中
sort算法对deque容器中分数排序,去除最高和最低分
deque容器遍历一遍,累加总分
获取平均分*/

class Person
{
public:
	Person(int score, string name)
	{
		m_name = name;
		m_score = score;
	}
	int m_score;
	string m_name;
};
void createPerson(vector<Person>&v)
{
	string nameSeed = "ABCDE";
	for (int i = 0; i < 5; i++)
	{
		string name = "player_";
		name += nameSeed[i];
		int score = 0;
		Person p(score, name);
		v.push_back(p);
	}
}
void setScore(vector<Person>&v)
{
	for (vector<Person>::iterator it = v.begin(); it != v.end(); it++)
	{
		deque<int>d;
		for (int i = 0; i < 10; i++)
		{
			int score = rand() % 51 + 50;   //int score = [rand() % 51] + 50; []内取值为0~50
			d.push_back(score);
		}
		sort(d.begin(),d.end());
		d.pop_back();
		d.pop_front();
		int sum = 0;
		for (deque<int>::iterator dit = d.begin(); dit != d.end(); dit++)
		{
			sum += *dit;
		}
		int avg = sum / d.size();
		it->m_score = avg;
	}
}

void func()     
{
	vector<Person>p;
	createPerson(p);
	setScore(p);
	for (vector<Person>::iterator it = p.begin(); it != p.end(); it++)
	{
		cout << it->m_name << "的最终得分为" << it->m_score << endl;
	}
}

int main() {

	srand((unsigned int)time(NULL));
	func();

	system("pause");

	return 0;
}

五、stack容器

1、stack的基本概念

(1)stack是一种先进后出(First In Last Out,FILO)的数据结构,它只有一个出口。

(2)栈中只有顶端的元素才可以被外界使用,因此栈不允许有遍历行为。

(1)栈中进入数据的过程称为入栈(push),栈中弹出数据的过程称为出栈(pop)。

2、stack的常用接口

//构造函数:

stack<T> stk;          //stack采用模板类实现,stack对象的默认构造形式

stack(const stack &stk);  //拷贝构造函数

//赋值操作:

stack& operator=(const stack &stk);   //重载等号操作符

//数据存取:

push(elem);   //向栈顶添加元素

pop();        //从栈顶移除第一个元素

top();        //返回栈顶元素

//大小操作:

empty();      //判断堆栈是否为空

size();        //返回栈的大小

#include<iostream>
#include<stack>
using namespace std;

void test01()     
{
	stack<int>s;    //符合先进后出的数据结构

	s.push(10);     //入栈
	s.push(20);
	s.push(30);
	s.push(40);

	while (!s.empty())   //只要栈不为空,查看栈顶,并执行出栈操作
	{
		cout << "栈顶元素为:" << s.top() << endl;
		s.pop();    //出栈
	}
	cout << "现在栈的大小为:" << s.size() << endl;
}

int main() {

	test01();

	system("pause");

	return 0;
}

六、queue 容器

1、queue的基本概念

(1)Queue是一种先进先出(First In First Out,FIFO)的数据结构,它有两个出口。

(2)队列容器允许从一端(队尾)新增元素,从另一端(队头)移除元素。

(3)队列中只有队头和队尾才可以被外界使用,因此队列不允许有遍历行为。

(4)数据进入队列中的过程称为入队(push),数据从队列中被移出的过程称为出队(pop)。

2、queue的常用接口

//构造函数:

queue<T> que;             //queue采用模板类实现,queue对象的默认构造形式

queue(const queue &que);   //拷贝构造函数

//赋值操作:

queue& operator=(const queue &que); //重载等号操作符

//数据存取:

push(elem);   //往队尾添加元素

pop();        //从队头移除第一个元素

back();       //返回最后一个元素

front();       //返回第一个元素

//大小操作:

empty();     //判断堆栈是否为空

size();       //返回栈的大小

#include<iostream>
#include<queue>
using namespace std;

class Person
{
public:
	int age1;
	int age2;
	Person(int age1, int age2)
	{
		this->age1 = age1;
		this->age2 = age2;
	}
};

void test01()     
{
	queue<Person>q;       //创建队列
	Person p1(10, 18);    //准备数据
	Person p2(20, 38);
	Person p3(30, 58);
	Person p4(40, 28);
	q.push(p1);           //入队
	q.push(p2);
	q.push(p3);
	q.push(p4);
	cout << "队列大小为:" << q.size() << endl;
	while (!q.empty())
	{
		cout << "队头为: age1 = " << q.front().age1 << "  age2 = " << q.front().age2 << endl;
		cout << "队尾为: age1 = " << q.back().age1 << "  age2 = " << q.back().age2 << endl;
		cout << endl;
		q.pop();          //出队
	}
	cout << "队列大小为:" << q.size() << endl;
}

int main() {

	test01();

	system("pause");

	return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/400522.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

接口自动化测试利器,使用Rest Assured进行REST API测试

我们在做接口测试时&#xff0c;一般在代码中会使用HttpClient&#xff0c;但是HttpClient相对来讲还是比较麻烦的&#xff0c;代码量也相对较多&#xff0c;对于新手而言上手会比较难一点&#xff0c;今天我们来看下另一个接口测试工具包REST Assured REST Assured是一个流行…

Qt 基础之进度条 - QProgressDialog和QProgressBar

Qt 基础之进度条 - QProgressDialog和QProgressBar 引言一、QProgressDialog例程1.1 效果展示1.2 源码 二、QProgressBar例程2.1 效果展示2.2 源码 三、QProgressBar进阶 引言 进度条的作用是用于显示任务或操作的进度&#xff0c;以便用户了解当前任务的完成情况。它可以提供…

如何删除PS最近使用项

ps删除最近文件列表 点击菜单栏中文件&#xff0d;>最近打开文件&#xff0d;>清除最近的文件列表

【python】windowslinux系统python的安装

一、python官网及下载路径 官网地址&#xff1a;Welcome to Python.org 下载路径&#xff1a;Download Python | Python.org ​​​​​​​ linux源码安装包下载&#xff1a; windows二进制安装包下载&#xff1a; 二、Linux如何安装python 2.1 单版本安装 以安装python…

Python实现线性逻辑回归和非线性逻辑回归

线性逻辑回归 # -*- coding: utf-8 -*- """ Created on 2024.2.20author: rubyw """import matplotlib.pyplot as plt import numpy as np from sklearn.metrics import classification_report from sklearn import preprocessing from sklearn…

Java+SpringBoot+Vue的大学生就业信息管理系统

一、项目介绍 基于Java (spring-boot)的大学生就业信息管理系统分为三个角色&#xff1a;管理员、企业、求职者。 功能:登录、注册功能、学生信息管理、企业信息管理、岗位分类管理、学历信息管理、应聘信息管理、求职者信息管理、招聘信息管理。 二、作品包含 三、项目技术 后…

BERT架构简介

一、BERT模型架构 BERT沿用原始Transformer模型中的编码器层&#xff0c;具有编码器的堆叠。但BERT没有使用解码器层&#xff0c;因此没有掩码多头注意力子层。&#xff08;BERT的设计者认为&#xff0c;对序列后续部分进行掩码会阻碍注意力过程&#xff09;。于是&#xff0c;…

Day 30 标准IO

文章目录 1.什么是标准IO1.1 概念1.2 特点1.3 操作 2.缓存区3.函数接口3.1 打开文件fopen3.2 关闭文件 fclose3.3 读写文件操作3.3.1 每次读写一个字符&#xff1a;fgetc()、fputc()每次读一个字符fgetc()每次写一个字符fputc()(1)针对文件(2)针对终端feof和ferror 3.3.2 每次一…

笔试题详解(C语言进阶)

前言 欢迎阅读本篇文章&#xff01;本篇文章通过一个笔试题来加强我们对C语言的理解&#xff0c;希望对你有帮助。后续我会写一个栏目&#xff0c;集合我见到的C语言题目&#xff0c;进行分析讲解。 1、题目一 判断下面程序的输出结果&#xff1a;(下面说的地址4/8字节是因为对…

【GameFramework框架内置模块】2、数据节点(Data Node)

推荐阅读 CSDN主页GitHub开源地址Unity3D插件分享简书地址 大家好&#xff0c;我是佛系工程师☆恬静的小魔龙☆&#xff0c;不定时更新Unity开发技巧&#xff0c;觉得有用记得一键三连哦。 一、前言 【GameFramework框架】系列教程目录&#xff1a; https://blog.csdn.net/q7…

机器学习 day38(有放回抽样、随机森林算法、XGBoost)

有放回抽样 有放回抽样和无放回抽样的区别&#xff1a;有放回可以确保每轮抽取的结果不一定相同&#xff0c;无放回则每轮抽取的结果都相同 在猫狗的例子中&#xff0c;我们使用”有放回抽样“来抽取10个样本&#xff0c;并组合为一个与原始数据集不同的新数据集&#xff0c;虽…

ORM中常用的字段和参数,正反向概念

django表查询测试环境搭建 首先&#xff0c;在此之前我们先来回顾一下之前学习的orm内容 1. django自带一个小型的sqlite3的小型数据库 但是这个数据库的功能非常有限&#xff0c;并且针对日期类型的数据兼容性很差 2. 切换数据库数据(MySQL) 2.1 在django1.x版本中你需要在_…

盲水印、暗水印(Blind Watermark)算法简明教程:算法原理、流程以及基于C/C++ 的代码实现

前言 由于工作需要&#xff0c;最近学习了盲水印相关的知识&#xff0c;本文对学习过程中做一个整理和总结。主要内容包括&#xff1a; 对盲水印相关概念做基本介绍对开源的 python 算法 blind_watermark 进行解析&#xff0c;给出算法流程基于 blind_watermark&#xff0c;给…

qt - 19种精美软件样式

qt - 19种精美软件样式 一、效果演示二、核心程序三、下载链接 一、效果演示 二、核心程序 #include "mainwindow.h"#include <QtAdvancedStylesheet.h> #include <QmlStyleUrlInterceptor.h>#include "ui_mainwindow.h" #include <QDir&g…

提高工作效率,体验ONLYOFFICE办公软件

ONLYOFFICE办公软件 一、前言二、特点完整办公套件协作过程更容易 三、访问地址 一、前言 随着数字化办公的普及&#xff0c;办公软件在我们的工作中扮演着越来越重要的角色。为了提高工作效率&#xff0c;我们需要一个功能强大、易于使用的办公软件。ONLYOFFICE作为一款全功能…

如何通过SQL语句获取表/视图的DDL,表/列/索引的统计信息,查询的执行计划(MySQL)

文章目录 获取对象的定义SQL语句列出库中的表和视图表的DDL语句索引的DDL语句视图的DDL语句物化视图的DDL语句 获取统计信息的SQL语句表级统计信息索引统计信息列级统计信息 获取执行计划的Explain语句ExplainExplain JsonExplain Tree (8.0.16及以上)Explain Analyze (8.0.18及…

Java语言实现五子棋

目录 内容 题目 解题 代码 实现 内容 题目 五子棋 使用二维数组,实现五子棋功能. 1.使用二维数组存储五子棋棋盘 如下图 2.在控制台通过Scanner输入黑白棋坐标(例如:1,2 2,1格式 表示二维数组坐标),使用实心五角星和空心五角星表示黑白棋子. 如下图: 输入后重新输出…

解决方案——文本生成图像DF-GAN配置Oxford-102 Flower 花数据集全流程

目录 一、Oxford-102 Flower简介二、DF-GAN配置Oxford-102 Flower 数据集2.1、下载数据集2.2、配置数据集 三、修改代码四、资源下载 一、Oxford-102 Flower简介 Oxford-102 Flower是牛津工程大学于2008年发布的用于图像分类的花卉数据集&#xff0c;原论文链接&#xff1a;Au…

适配器模式:转换接口,无缝对接不同系统

文章目录 **一、技术背景与应用场景****为什么使用适配器模式&#xff1f;****典型应用场景包括但不限于&#xff1a;** **二、适配器模式定义与结构****三、使用步骤举例****四、优缺点分析****总结** 一、技术背景与应用场景 适配器模式在软件设计中扮演着桥梁角色&#xff…

.zip文件如何在centos7解压

在CentOS 7中解压.zip文件&#xff0c;您可以使用unzip命令。如果您的系统上没有安装unzip工具&#xff0c;您首先需要安装它。以下是步骤指南&#xff1a; 安装unzip工具 打开终端并运行以下命令来安装unzip&#xff1a; sudo yum install unzip解压.zip文件 安装unzip之后&am…