如何運用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;
}

希望能夠將自己的一些學習經驗分享給有需要的人。
我是小鄭,一個堅持不懈的小白。

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