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;
}
希望能夠將自己的一些學習經驗分享給有需要的人。
我是小鄭,一個堅持不懈的小白。