如何运用C++ set 附例题(set中erase函数、iterator迭代器)

set的头文件为< set >,提供了如下操作:

(1)s.begin() 返回指向第一个元素的迭代器。
(2)s.clear() 清除所有元素。
(3)s.count() 返回某个值元素的个数。
(4)s.empty() 如果集合为空,返回true(真)。
(5)s.end() 返回指向最后一个元素之后的迭代器,不是最后一个元素。
(6)s.erase() 删除集合中的元素。
(7)s.find() 返回一个指向被查找到元素的迭代器。
(8)s.insert() 在集合中插入元素。
(9)s.max_size() 返回集合能容纳的元素的最大限值。
(10)s.size() 集合中元素的数目。
(11)s.swap() 交换两个集合变量。
set放入一个元素就会调整这个元素的位置,把它放到合适的位置,所以set中只有一个insert插入操作,而且set不能保存相同的元素。

例一:日期问题
set 迭代器 iterator:按从小到大读取set中元素。

#include<set>
#include<iostream>
using namespace std;
set<int> s;
int main(){
set< int >::iterator iter;
s.insert(50);
s.insert(100);
for(iter = s.begin(); iter != s.end(); iter++)
{
	cout<<*iter<<" ";
}
}

小明正在整理一批历史文献。这些历史文献中出现了很多日期。小明知道这些日期都在1960年1月1日至2059年12月31日。令小明头疼的是,这些日期采用的格式非常不统一,有采用年/月/日的,有采用月/日/年的,还有采用日/月/年的。更加麻烦的是,年份也都省略了前两位,使得文献上的一个日期,存在很多可能的日期与其对应。

比如02/03/04,可能是2002年03月04日、2004年02月03日或2004年03月02日。

给出一个文献上的日期,你能帮助小明判断有哪些可能的日期对其对应吗?

输入
一个日期,格式是"AA/BB/CC"。 (0 <= A, B, C <= 9)
输出
输出若干个不相同的日期,每个日期一行,格式是"yyyy-MM-dd"。多个日期按从早到晚排列。

样例输入
02/03/04
样例输出
2002-03-04
2004-02-03
2004-03-02


题意:在给定的文献日期上,判断有哪些可能的日期对其对应。
这道题主要考查细节
①、考虑年份是否为闰年
②、不同月份的长度
③、若有多个可能日期,则按从早到晚排列

set 迭代器 iterator:排序、去重

#include<bits/stdc++.h>
using namespace std;
char s[10];
int run[15]={0, 31, 29, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31};
int fr[15]={0, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31};

i2s(int i, string& s)
{
	stringstream ss;
	ss<<i;
	ss>>s;
}

string f(int a, int b, int c)
{
	//预处理 
	a<=59?a+=2000:a+=1900;
	string _a, _b, _c;
	i2s(a, _a);	i2s(b, _b);	i2s(c, _c);
	if(_b.length()==1) _b = "0" + _b;
	if(_c.length()==1) _c = "0" + _c;
	
	//判断是否合法 
	if(((a%4==0 && a%100!=0) || a%400==0 ) && b!=0 && b<=12
		&& c!=0 && c<=run[b])
		return _a+"-"+_b+"-"+_c;
	else if(b!=0&&b<=12&&c!=0&&c<=fr[b])
		return _a+"-"+_b+"-"+_c;
	
	return "";
} 
int main()
{
	//读取输入 
	int a, b, c;
	cin.getline(s, 10);
	sscanf(s, "%d/%d/%d", &a, &b, &c);
	
	//生成字符串 
	string cases[3];
	cases[0] = f(a, b, c);
	cases[1] = f(c, a, b);
	cases[2] = f(c, b, a);
	
	//字符串排序:set实现自动排序且去重 
	set<string> ans;
	for(int i=0; i<3; i++)
		if(cases[i] != "")
			ans.insert(cases[i]);
	set<string>::iterator iter;		//迭代器:输出排序字符串 
	for(iter=ans.begin(); iter!=ans.end(); iter++)
		cout<<*iter<<endl;
	
	return 0;
}

在set中,erase()函数十分便捷,能够删除set容器中指定的元素,不像栈只能读取栈顶元素。

set.erase():删除set容器中指定元素。
eg:
set< int > s;
s.erase(50); //删除s中值为50的元素

例二:木材仓库
题目描述
博艾市有一个木材仓库,里面可以存储各种长度的木材,但是保证没有两个木材的长度是相同的。作为仓库负责人,你有时候会进货,有时候会出货,因此需要维护这个库存。有不超过 100000 条的操作:

  • 进货,格式1 Length:在仓库中放入一根长度为 Length(不超过 109 ) 的木材。如果已经有相同长度的木材那么输出Already Exist。
  • 出货,格式2 Length:从仓库中取出长度为 Length 的木材。如果没有刚好长度的木材,取出仓库中存在的和要求长度最接近的木材。如果有多根木材符合要求,取出比较短的一根。输出取出的木材长度。如果仓库是空的,输出Empty。

输入输出样例
输入
7
1 1
1 5
1 3
2 3
2 3
2 3
2 3
输出
3
1
5
Empty


map:记录不同长度的木材是否存入仓库。
set:仓库中的木材按照不同长度进行排序,方便取木材。
具体解题思路请看代码注释

#include<bits/stdc++.h>
#define ll long long
using namespace std;
map <ll, ll> m;
set <ll> s;

int main(){
	ll n, count=0;
	cin>>n;
	set<ll>::iterator iter;
	while(n--){
		ll a, b;
		cin>>a>>b;
		//存货 
		if(a==1){
			//已存在 
			if(m[b]==1){
				cout<<"Already Exist";
				if(n!=0){
					cout<<endl;
				}
			}
			//不存在 
			else if(m[b]==0){
				m[b]=1;
				s.insert(b);
				count++;
			}
		}
		//取货 
		else if(a==2){
			//空仓 
			if(count==0){
				cout<<"Empty";
			}
			//有货 
			else if(m[b]==1){
				m[b]=0;
				s.erase(b);
				count--;
				cout<<b;
			}
			//没确定长度的货 
			else if(m[b]==0){
				ll ans=0;
				//找货 
				for(iter=s.begin(); iter!=s.end(); iter++){
					//偏小的货 
					if(*iter<b){
						ans=*iter;
					}
					//偏大的货 
					else if(*iter>b){
						//如果偏大的货较偏小的货长度接近想要的货
						//拿偏大的货 
						if(abs(b-ans)>abs(*iter-b)||ans==0){
							ans = *iter;
						}
					}
				}
				m[ans]=0;
				s.erase(ans);
				count--;
				cout<<ans;
			}
			if(n!=0){
				cout<<endl;
			}
		}
	}
	
	return 0;
}

希望能够将自己的一些学习经验分享给有需要的人。
我是小郑,一个坚持不懈的小白。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章