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;
}
希望能够将自己的一些学习经验分享给有需要的人。
我是小郑,一个坚持不懈的小白。