C++實(shí)現(xiàn)std::set的示例項(xiàng)目
std::set
是 C++ 標(biāo)準(zhǔn)庫中的關(guān)聯(lián)容器,它提供了一種存儲唯一元素的有序集合。它提供了高效的插入、刪除和查找操作,并且能夠自動維護(hù)元素的有序性和唯一性。
一、底層實(shí)現(xiàn)
std::set
的底層原理是基于紅黑樹
(Red-Black Tree)。紅黑樹是一種自平衡的二叉搜索樹,保證了樹的平衡性,從而保證了插入、刪除和查找操作的時間復(fù)雜度為 O(log n)。它滿足以下性質(zhì):
1、 每個節(jié)點(diǎn)要么是紅色,要么是黑色。
2、 根節(jié)點(diǎn)是黑色的。(插入的第一個元素)
3、 所有葉子節(jié)點(diǎn)(NIL 節(jié)點(diǎn))都是黑色的。
4、 如果一個節(jié)點(diǎn)是紅色的,則它的兩個子節(jié)點(diǎn)都是黑色的。
5、 對于每個節(jié)點(diǎn),從該節(jié)點(diǎn)到其后代葉子節(jié)點(diǎn)的簡單路徑上,黑色節(jié)點(diǎn)的數(shù)量都相同。
在 std::set
中,每個元素都被 插入到紅黑樹中作為一個節(jié)點(diǎn)。當(dāng)插入一個新元素時,std::set
會根據(jù)元素值的大小將其插入到合適的位置,同時保持紅黑樹的平衡性。在紅黑樹中查找元素時,根據(jù)紅黑樹的特性,可以通過二分查找的方式高效地定位到目標(biāo)元素。
由于紅黑樹是一種高效的自平衡二叉搜索樹,因此 std::set
能夠提供高效的插入、刪除和查找操作,同時保持元素的有序性和唯一性。
二、成員函數(shù)
std::set提供了一系列成員函數(shù)來操作集合。
insert
: 向集合中插入元素。emplace
: 在集合中就地構(gòu)造元素。erase
: 從集合中刪除指定元素。find
: 查找集合中是否存在指定元素。count
: 返回集合中與指定元素相等的元素的數(shù)量(通常為0或1)。clear
: 清空集合中的所有元素。size
: 返回集合中元素的數(shù)量。empty
: 檢查集合是否為空。swap
: 交換兩個集合的內(nèi)容。begin
: 返回指向集合中第一個元素的迭代器。end
: 返回指向集合中最后一個元素之后位置的迭代器。lower_bound
: 返回指向第一個不小于給定鍵值的元素的迭代器。upper_bound
: 返回指向第一個大于給定鍵值的元素的迭代器。equal_range
: 返回一個范圍,其中包含所有與給定鍵值相等的元素。max_size
: 返回集合能容納的最大元素?cái)?shù)量。
Demo演示:
std::set<int> set1; set1.insert(1); set1.insert(2); set1.insert(3); set1.insert(4); std::cout<<set1.size()<<std::endl; // 4 std::cout<<set1.empty()<<std::endl; // 0 std::cout<<*set1.find(2)<<std::endl; // 2 set1.erase(2); std::cout<<(set1.find(2)==set1.end())<<std::endl; // 1 std::cout<<set1.count(3)<<std::endl; // 1 auto lower = set1.lower_bound(3); auto upper = set1.upper_bound(3); std::cout << "Lower bound of 3: " << *lower << std::endl; // Lower bound of 3: 3 std::cout << "Upper bound of 3: " << *upper << std::endl; // Upper bound of 4: 4 std::set<int> set2; set2.swap(set1); std::cout<<set1.size()<<" "<<set2.size()<<std::endl; // 0 3 std::cout<<set1.max_size()<<std::endl; // 230584300921369395 for (auto it = set2.begin(); it != set2.end(); ++it) { std::cout << *it << " "<<std::endl; // 1 3 4 } auto range = set2.equal_range(3); std::cout<<*range.first<<std::endl; // 3
三、自定義排序規(guī)則
std::set
默認(rèn)是根據(jù)元素的大小進(jìn)行排序的,采用的是嚴(yán)格弱序(Strict Weak Ordering)的比較方式。這意味著元素必須支持 <
運(yùn)算符,元素按照升序排列。
可以通過提供自定義的比較函數(shù)或者函數(shù)對象來實(shí)現(xiàn)排序規(guī)則。需要兩個步驟:
1、 定義比較函數(shù)或者函數(shù)對象:
比較函數(shù):定義一個函數(shù),接受兩個參數(shù),比較它們的大小,并返回 true
或 false
,表示它們的順序關(guān)系。
函數(shù)對象(Functor):定義一個類,重載 operator()
,使得對象可以像函數(shù)一樣被調(diào)用,在其中實(shí)現(xiàn)比較邏輯。
2、 傳遞比較函數(shù)或者函數(shù)對象給 std::set
構(gòu)造函數(shù):
在創(chuàng)建 std::set
對象時,通過傳遞比較函數(shù)或者函數(shù)對象作為第二個參數(shù),告訴集合如何比較元素的順序。
#include <iostream> #include <set> // 自定義比較函數(shù) bool customCompare(int a, int b) { // 實(shí)現(xiàn)自定義的比較邏輯,這里以整數(shù)的絕對值大小為例 return abs(a) < abs(b); } // 自定義函數(shù)對象(Functor) struct CustomComparator { bool operator()(int a, int b) const { // 實(shí)現(xiàn)自定義的比較邏輯,這里以整數(shù)的平方大小為例 return (a * a) < (b * b); } }; int main() { // 使用自定義比較函數(shù)創(chuàng)建 std::set std::set<int, decltype(&customCompare)> customSet(customCompare); // 使用自定義函數(shù)對象創(chuàng)建 std::set std::set<int, CustomComparator> functorSet; // 插入元素 customSet.insert(5); customSet.insert(-3); customSet.insert(2); functorSet.insert(5); functorSet.insert(-3); functorSet.insert(2); // 遍歷輸出結(jié)果 std::cout << "Custom compare function set: "; for (int num : customSet) { std::cout << num << " "; } std::cout << std::endl; std::cout << "Functor set: "; for (int num : functorSet) { std::cout << num << " "; } std::cout << std::endl; return 0; }
四、自定義對象排序
當(dāng) std::set
中存儲的元素是自定義對象時,需要重載 <
運(yùn)算符,以便 std::set
能夠?qū)ο筮M(jìn)行比較和排序。重載 <
運(yùn)算符的目的是定義對象之間的比較規(guī)則,以確定它們在集合中的順序。這樣,std::set
就可以使用這些規(guī)則來維護(hù)集合的有序性。
下面是一個示例,演示了如何定義一個自定義類型的 Person
對象,并在 std::set
中使用自定義的比較函數(shù)進(jìn)行排序:
#include <iostream> #include <set> #include <string> // 定義一個自定義的 Person 類 class Person { private: std::string name; int age; public: Person(std::string name, int age) : name(name), age(age) {} // 重載 < 運(yùn)算符,用于自定義對象的比較規(guī)則 bool operator<(const Person& other) const { // 按照年齡進(jìn)行比較 return age < other.age; } // 為了便于輸出,重載 << 運(yùn)算符 friend std::ostream& operator<<(std::ostream& os, const Person& person) { os << "Name: " << person.name << ", Age: " << person.age; return os; } }; int main() { // 創(chuàng)建一個存儲 Person 對象的 std::set 容器,并使用自定義的比較函數(shù)進(jìn)行排序 std::set<Person> people; // 向集合中插入一些 Person 對象 people.insert(Person("Alice", 30)); people.insert(Person("Bob", 25)); people.insert(Person("Charlie", 35)); // 遍歷輸出集合中的元素,由于使用了自定義的比較規(guī)則,輸出時會按照年齡升序排列 for (const auto& person : people) { std::cout << person << std::endl; } return 0; }
在這個示例中,Person
類具有 name
和 age
兩個屬性,我們通過重載 <
運(yùn)算符來定義了對象之間的比較規(guī)則,按照年齡進(jìn)行比較。然后,我們創(chuàng)建了一個 std::set<Person>
容器,并向其中插入了幾個 Person
對象。由于我們使用了自定義的比較函數(shù),因此在輸出容器中的元素時,它們會按照年齡的升序排列。
五、性能問題
std::set
在插入、刪除和查找等操作上具有穩(wěn)定且良好的性能特征,適用于需要高效管理有序唯一元素集合的場景。然而,在空間效率方面,它可能不如其他數(shù)據(jù)結(jié)構(gòu)(如 std::unordered_set
)那么優(yōu)越。因此,在選擇數(shù)據(jù)結(jié)構(gòu)時,需要根據(jù)具體的需求權(quán)衡其性能特征。
1、 插入和刪除操作的性能:
插入和刪除操作的時間復(fù)雜度為 O(log n),其中 n 是集合中的元素?cái)?shù)量。這是因?yàn)榧t黑樹是一種自平衡的二叉搜索樹,插入和刪除操作會導(dǎo)致樹的重新平衡,但是由于紅黑樹的特性,重新平衡的代價是受控的。
因此,std::set
在插入和刪除方面的性能比較穩(wěn)定,不受集合大小的影響。
2、 查找操作的性能:
查找操作的時間復(fù)雜度也是 O(log n),因?yàn)榧t黑樹具有良好的平衡性,可以保證在平均情況下快速地查找元素。std::set
提供了高效的查找功能,適用于需要快速檢索元素的場景。
3、 迭代器的性能:std::set
的迭代器是雙向迭代器,支持前向和后向遍歷,其性能與集合大小無關(guān),是常數(shù)時間復(fù)雜度的操作。
這意味著在遍歷 std::set
中的元素時,無論集合大小如何,迭代器的操作都非常高效。
4、 內(nèi)存占用:std::set
使用紅黑樹作為底層數(shù)據(jù)結(jié)構(gòu),它是一種高效的平衡二叉搜索樹,但是相對于其他數(shù)據(jù)結(jié)構(gòu)(如哈希表),它可能會占用更多的內(nèi)存。
紅黑樹需要維護(hù)額外的節(jié)點(diǎn)信息以保持平衡,因此在存儲相同數(shù)量的元素時,std::set
可能會占用更多的內(nèi)存空間。
到此這篇關(guān)于C++實(shí)現(xiàn)std::set的示例項(xiàng)目的文章就介紹到這了,更多相關(guān)C++ std::set內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- C++中std::setw()的用法解讀
- c++中std::placeholders的使用方法
- c++ std::sort使用自定義的比較函數(shù)排序方式
- C++中std::thread{}和std::thread()用法
- C++中std::tuple和std::pair的高級用法
- c++之std::get_time和std::put_time
- C++中std::ios_base::floatfield報錯已解決
- C++中std::invalid_argument報錯解決
- C++中std::ifstream::readsome和std::ifstream::read的區(qū)別解析
- C++中的std::funture和std::promise實(shí)例詳解
- C++中std::transform的使用小結(jié)
- C++?std::copy與memcpy區(qū)別小結(jié)
相關(guān)文章
C++中獲取隨機(jī)數(shù)的常用方法小結(jié)
這篇文章主要為大家詳細(xì)介紹了C++中獲取隨機(jī)數(shù)的幾種常用方法,文中的示例代碼講解詳細(xì),具有一定的借鑒價值,感興趣的小伙伴可以了解下2025-01-01使用C/C++讀取matlab中.mat格式數(shù)據(jù)的操作
這篇文章給大家介紹了使用C/C++讀取matlab中.mat格式數(shù)據(jù)的操作,文中通過圖文結(jié)合的方式介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作有一定的幫助,需要的朋友可以參考下2023-12-12c語言實(shí)現(xiàn)通訊錄管理系統(tǒng)詳細(xì)實(shí)例
這篇文章主要給大家介紹了關(guān)于c語言實(shí)現(xiàn)通訊錄管理系統(tǒng)的相關(guān)資料,通訊錄管理系統(tǒng)是一種常見的應(yīng)用程序,可以用來管理聯(lián)系人的信息,包括姓名、電話號碼、地址等,需要的朋友可以參考下2023-07-07