c++ STL 學習筆記(pair 和 set)
STL pair
(1)pair 的定義
標頭檔案 <utility>
STL的標頭檔案中描述了一個看上去非常簡單的模版類pair,用來表示一個二元組或元素對,並提供了按照字典序對元素對進行大小比較運算子模版函式。
定義一個pair物件表示一個平面座標點:
例:
pair<double, double> p;
cin >> p.first >> p.second;
或者
pair <string,double> product1 ("tomatoes",3.25);
pair模版類需要兩個引數:首元素的資料型別和尾元素的資料型別。pair模版類物件有兩個成員:first和second,分別表示首元素和尾元素。
例:
pair<int,int>p1; pair<int,int>p2; cin>>p1.first>>p1.second; cin>>p2.first>>p2.second; cout<<'<'<<p1.first<<','<<p1.second<<'>'<<endl; cout<<'<'<<p2.first<<','<<p2.second<<'>'<<endl;
輸出結果
<p1.first,p1.second>
<p2.first,p2.second>
(2)pair 的比較
在<utility>
中已經定義了pair上的六個比較運算子:<、>、<=、>=、==、!=,其規則是先比
較first,first相等時再比較second,這符合大多數應用的邏輯。當然,也可以通過過載這幾個運算子
來重新指定自⼰的比較邏輯。
例:
if(p1>p2) cout<<"p1"<<endl; else if(p2>p1) cout<<"p2"<<endl; else cout<<"equal"<<endl;
(3)pair 的即時生成
除了直接定義一個pair物件外,如果需要即時生成一個pair物件,也可以呼叫在<utility>
中定
義的一個模版函式:make_pair。make_pair需要兩個引數,分別為元素對的首元素和尾元素。
例:
pair <string,double> product3;
product3 = make_pair ("shoes",20.0);
cout <<"The price of "<< product3.first <<" is $"<< product3.second <<"\n";
輸出結果:
The price of shoes is $ 20.0
注:
一般make_pair都使用在需要pair做引數的位置,可以直接呼叫make_pair生成pair物件。
另一個使用的方面就是pair可以接受隱式的型別轉換,這樣可以獲得更高的靈活度。但是這樣會出現如下問題:
例如有如下兩個定義:
std::pair<int, float>(1, 1.1);
std::make_pair(1, 1.1);
其中第一個的second變數是float型別,而make_pair函式會將sec新增連結描述ond變數都轉換成double型別。
這個問題在程式設計是需要引起注意。
STL set
引用自 c++set 用法詳解
(1)set 的定義:
標頭檔案<set>
關於set,必須說明的是set關聯式容器。set作為一個容器也是用來儲存同一資料型別的資料型別,並且能從一個數據集合中取出資料,在set中每個元素的值都唯一,而且系統能根據元素的值自動進行排序。應該注意的是set中數元素的值不能直接被改變。C++ STL中標準關聯容器set, multiset, map, multimap內部採用的就是一種非常高效的平衡檢索二叉樹:紅黑樹,也成為RB樹(Red-Black Tree)。RB樹的統計效能要好於一般平衡二叉樹,所以被STL選擇作為了關聯容器的內部結構。
定義一個set物件:
set<int> s;
set<double> ss;
(2)set 的基本操作:
s.begin() // 返回指向第一個元素的迭代器
s.clear() // 清除所有元素
s.count() // 返回某個值元素的個數
s.empty() // 如果集合為空,返回true(真)
s.end() // 返回指向最後一個元素之後的迭代器,不是最後一個元素
s.equal_range() // 返回集合中與給定值相等的上下限的兩個迭代器
s.erase() // 刪除集合中的元素
s.find() // 返回一個指向被查詢到元素的迭代器
s.get_allocator() // 返回集合的分配器
s.insert() // 在集合中插入元素
s.lower_bound() // 返回指向大於(或等於)某值的第一個元素的迭代器
s.key_comp() // 返回一個用於元素間值比較的函式
s.max_size() // 返回集合能容納的元素的最大限值
s.rbegin() // 返回指向集合中最後一個元素的反向迭代器
s.rend() // 返回指向集合中第一個元素的反向迭代器
s.size() // 集合中元素的數目
s.swap() // 交換兩個集合變數
s.upper_bound() // 返回大於某個值元素的迭代器
s.value_comp() // 返回一個用於比較元素間的值的函式
例:
#include <iostream>
#include <set>
using namespace std;
set<int>s;
s.insert(1);
s.insert(2);
s.insert(4);
s.insert(0);
cout<<"set 的 size 值為 :"<<s.size()<<endl;
cout<<"set 的 maxsize的值為 :"<<s.max_size()<<endl;
cout<<"set 中的第一個元素是 :"<<*s.begin()<<endl;
cout<<"set 中的最後一個元素是:"<<*s.end()<<endl;
if(s.empty())
{
cout<<"set 為空 !!!"<<endl;
}
else cout<<"set 的 size 值為 :"<<s.size()<<endl;
s.clear();
if(s.empty())
{
cout<<"set 為空 !!!"<<endl;
}
else cout<<"set 的 size 值為 :"<<s.size()<<endl;
cout<<"set 的 size 值為 :"<<s.size()<<endl;
cout<<"set 的 maxsize的值為 :"<<s.max_size()<<endl;
return 0;
}
執行結果
set 的 size 值為 :4
set 的 maxsize的值為 :214748364
set 中的第一個元素是 :0
set 中的最後一個元素是:4
set 的 size 值為 :4
set 為空 !!!
set 的 size 值為 :0
set 的 maxsize的值為 :214748364
rbegin()與rend()
反向迭代器是一種反向遍歷容器的迭代器。也就是,從最後一個元素到第一個元素遍歷容器。反向迭代器將自增(和自減)的含義反過來了:對於反向迭代器,++ 運算將訪問前一個元素,而 – 運算則訪問下一個元素。
.begin() 返回一個迭代器,它指向容器c的第一個元素
.end() 返回一個迭代器,它指向容器c的最後一個元素的下一個位置
.rbegin() 返回一個逆序迭代器,它指向容器c的最後一個元素
.rend() 返回一個逆序迭代器,它指向容器c的第一個元素前面的位置
反向迭代器用於表示範圍,而所表示的範圍是不對稱的,這個事實可推匯出一個重要的結論:使用普通的迭代器對反向迭代器進行初始化或賦值時,所得到的迭代器並不是指向原迭代器所指向的元素。
set的遍歷
#include<cstdio>
#include<set>
using namespace std;
int main()
{
set<int>s;
s.insert(3);
s.insert(1);
s.insert(2);
s.insert(1);
set<int>::iterator it;
for(it=s.begin();it!=s.end();it++) //使用迭代器進行遍歷
{
printf("%d\n",*it);
}
return 0;
}
//輸出結果 : 1 2 3 一共插入了4個數,但是集合中只有3個數並且是有序的,可見之前說過的set集合的兩個特點,有序和不重複。
小結: 還要注意begin() 和 end()函式是不檢查set是否為空的,使用前最好使用empty()檢驗一下set是否為空.
count()
用來查詢set中某個某個鍵值出現的次數。這個函式在set並不是很實用,因為一個鍵值在set只可能出現0或1次,這樣就變成了判斷某一鍵值是否在set出現過了。
例:
#include <iostream>
#include <set>
using namespace std;
int main(){
set<int> s;
s.insert(1);
s.insert(2);
s.insert(3);
s.insert(1);
cout<<"set 中 1 出現的次數是 :"<<s.count(1)<<endl;
cout<<"set 中 4 出現的次數是 :"<<s.count(4)<<endl;
return 0;
}
結果
set 中 1 出現的次數是 : 1
set 中 4 出現的次數是 : 0
equal_range()
返回一對定位器,分別表示第一個大於或等於給定關鍵值的元素和 第一個大於給定關鍵值的元素,這個返回值是一個pair型別,如果這一對定位器中哪個返回失敗,就會等於end()的值。
例:
#include <iostream>
#include <set>
using namespace std;
int main()
{
set<int>s;
set<int>::iterator ter;
for(int i=1;i<5;i++)
s.insert(i);
pair<set<int>::const_iterator,set<int>::const_iterator> pr;
pr = s.equal_range(3);
cout<<"第一個大於等於 3 的數是 :"<<*pr.first<<endl;
cout<<"第一個大於 3的數是 : "<<*pr.second<<endl;
return 0;
}
結果
第一個大於等於 3 的數是 : 3
第一個大於 3的數是 : 4
erase()
erase(iterator) ,刪除定位器iterator指向的值
erase(first,second),刪除定位器first和second之間的值
erase(key_value),刪除鍵值key_value的值
例:
#include <iostream>
#include <set>
using namespace std;
int main(){
set<int> s;
set<int>::const_iterator iter;
set<int>::iterator first;
set<int>::iterator second;
for(int i = 1 ; i <= 10 ; ++i)
{
s.insert(i);
}
for(iter = s.begin() ; iter != s.end() ; ++iter)
{
cout<<*iter<<" ";
}
cout<<endl;
//第一種刪除
s.erase(s.begin());
//第二種刪除
first=s.begin();
second=s.begin();
second++;
second++;
s.erase(first,second);
//第三種刪除
s.erase(8);
cout<<"刪除後 set 中元素是 :"<<endl;
for(iter = s.begin() ; iter != s.end() ; ++iter)
{
cout<<*iter<<" ";
}
cout<<endl;
return 0;
}
結果:
1 2 3 4 5 6 7 8 9 10
刪除後 set 中元素是 :
4 5 6 7 9 10
小結:set中的刪除操作是不進行任何的錯誤檢查的,比如定位器的是否合法等等,所以用的時候自己一定要注意。
insert()
insert(key_value)
將key_value插入到set中 ,返回值是pair<set::iterator,bool>,bool標誌著插入是否成功,而iterator代表插入的位置,若key_value已經在set中,則iterator表示的key_value在set中的位置。
inset(first,second)
將定位器first到second之間的元素插入到set中,返回值是void.
例:
#include <iostream>
#include <set>
using namespace std;
int main()
{
int a[] = {1,2,3};
int t;
set<int> s;
set<int>::iterator iter;
s.insert(a,a+3);
for(iter = s.begin() ; iter != s.end() ; ++iter)
{
cout<<*iter<<" ";
}
cout<<endl;
pair<set<int>::iterator,bool> pr;
cin>>t;
pr = s.insert(t);
if(!pr.second)
{
cout<<"error"<<endl;
}
for(iter = s.begin() ; iter != s.end() ; ++iter)
{
cout<<*iter<<" ";
}
cout<<endl;
return 0;
}
lower_bound&& upper_bound
lower_bound(key_value) ,返回第一個大於等於key_value的定位器
upper_bound(key_value),返回最後一個大於等於key_value的定位器
例:
#include <iostream>
#include <set>
using namespace std;
int main()
{
set<int> s;
s.insert(1);
s.insert(3);
s.insert(4);
cout<<*s.lower_bound(2)<<endl;
cout<<*s.lower_bound(3)<<endl;
cout<<*s.upper_bound(3)<<endl;
return 0;
}
結果:
3
3
4
.find()
find(Key)的功能是返回鍵值為Key的元素的位置,返回值是迭代器型別
例
#include <iostream>
#include <set>
using namespace std;
int main()
{
set<int>s;
for(int i=0;i<10;i++)
s.insert(i);
set<int>::iterator it1=s.find(4);
set<int>::iterator it2=s.find(11);
if(it1!=s.end())
cout<<*(it1)<<endl;
else cout<<"error"<<endl;
if(it2!=s.end())
cout<<*(it2)<<endl;
else cout<<"error"<<endl;
return 0;
}
結果
4
error
.swap()
進行兩個集合的交換
例
#include <iostream>
#include <set>
using namespace std;
int main()
{
set<int>s1;
set<int>s2;
for(int i=0;i<10;i++)
s1.insert(i);
for(int i=10;i<15;i++)
s2.insert(i);
cout<<"s1:";
for(set<int>::iterator it =s1.begin();it!=s1.end();it++)
cout<<*(it)<<" ";
cout<<endl;
cout<<"s2:";
for(set<int>::iterator it =s2.begin();it!=s2.end();it++)
cout<<*(it)<<" ";
cout<<endl;
s1.swap(s2);//進行交換
cout<<"s1:";
for(set<int>::iterator it =s1.begin();it!=s1.end();it++)
cout<<*(it)<<" ";
cout<<endl;
cout<<"s2:";
for(set<int>::iterator it =s2.begin();it!=s2.end();it++)
cout<<*(it)<<" ";
cout<<endl;
return 0;
}
結果
s1:0 1 2 3 4 5 6 7 8 9
s2:10 11 12 13 14
s1:10 11 12 13 14
s2:0 1 2 3 4 5 6 7 8 9
key_comp().value_comp()
在set中兩種函式使用方法相同
不過返回型別不同
key_comp()為key_compare型
value_comp()為value_compare型
例
#include <iostream>
#include <set>
#include <algorithm>
using namespace std;
int main()
{
set<double>s;
for(int i=0;i<10;i++)
s.insert(i);
set<double>::key_compare kc1=s.key_comp();
cout<<"s1:"<<endl;
for(set<double>::iterator it=s.begin();it!=s.end();it++)
{
cout<<*(it)<<" ";
if(kc1(7,*(it))==true) break;
}
cout<<endl;
return 0;
}
結果
s1:
0 1 2 3 4 5 6 7 8
自定義比較函式
(1)元素不是結構體:
例
//自定義比較函式myComp,過載“()”操作符
struct myComp
{
bool operator()(const your_type &a,const your_type &b)
{
return a.data-b.data>0;
}
}
set<int,myComp>s;
......
set<int,myComp>::iterator it;
(2)如果元素是結構體,可以直接將比較函式寫在結構體內。
例
struct Info
{
string name;
float score;
//過載“<”操作符,自定義排序規則
bool operator < (const Info &a) const
{
//按score從大到小排列
return a.score<score;
}
}
set<Info> s;
......
set<Info>::iterator it;
例
#include<stdio.h>
#include<set>
#include<string>
using namespace std;
struct People
{
string name;
int age;
bool operator <(const People p) const //運算子過載
{
return age<p.age; //按照年齡由小到大進行排序
}
};
int main()
{
set<People>s;
s.insert((People){"張三",14});
s.insert((People){"李四",16});
s.insert((People){"王二麻子",10});
set<People>::iterator it;
for(it=s.begin();it!=s.end();it++) //使用迭代器進行遍歷
{
printf("姓名:%s 年齡:%d\n",(*it).name.c_str(),(*it).age);
}
return 0;
}
結果
姓名:王二麻子 年齡:10
姓名:張三 年齡:14
姓名:李四 年齡:16
multiset是另一種型別的容器,其關鍵詞與資料檔案是同一個值,與set不同,其可包含重複的元素,其用法與set相似。