C++ 標準庫 sort() / stable_sort() / partial_sort() 對比

来源:https://www.cnblogs.com/zutterhao/archive/2023/03/23/17242437.html
-Advertisement-
Play Games

C++ STL標準庫中提供了多個用於排序的Sort函數,常用的包括有sort() / stable_sort() / partial_sort(),具體的函數用法如下表所示: | 函數 | 用法 | | | | | std::sort(first,last) | 對容器或數組first~last範圍 ...


C++ STL標準庫中提供了多個用於排序的Sort函數,常用的包括有sort() / stable_sort() / partial_sort(),具體的函數用法如下表所示:

函數 用法
std::sort(first,last) 對容器或數組first~last範圍內的元素進行排序,預設升序排序
std::stable_sort(first,last) 對容器或數組first~last範圍內的元素進行排序,保持原有數組相對順序,預設升序排序
std::partial_sort(first,middle,last) 在容器或數組first~last範圍內,查找最小(大)middle-first個元素排序,放入first-middle區間,預設升序

1. std::sort(first,last)

std::sort()是STL標準庫提供的模板函數,用於對容器或者數組中指定的範圍(first~last)元素進行排序,預設的排序方法是以元素的值的大小做升序排序,同時也可以指定其他的排序規則(如std::greater),也可以自定義排序規則。

std::sort()函數底層基於快速排序進行實現,時間複雜度為N * log(N),因此需要容器或者數組註意以下幾點:

  • 容器的迭代器必須是隨機訪問迭代器,如std::array、std::vector、std::deque。
  • 如果採用預設的升序排序方法,則元素必須支持operate<運算符。
  • 自定義類對象容器使用std::sort()排序時,因為需要交換位置,所以必須提供拷貝構造函數和賦值運算符。
  • std::sort()無法保證相同元素的原始位置,這裡的相同是指進行比較的結果為相等,而對象本身不相同。如果需要保持相對順序相同,則應該使用std::stable_sort()

不同庫對std::sort()的實現:libstdc++libc++.

示例代碼:

#include<iostream>
#include<vector>
#include<string>
#include<algorithm>

using namespace std;
void Print(std::string message,const std::vector<int>& vec)
{
    cout << message << ": ";
    for(const auto c : vec)
	cout << c << " ";
    cout <<endl;
}

int main()
{
    std::vector<int> myVector{5, 7, 4, 2, 8, 6, 1, 9, 0, 3};
    // 1. 以預設的排序規則
    std::sort(myVector.begin(),myVector.end());
    Print("Sorted Default", myVector);
    // 2. 以標準庫的greater函數進行排序
    std::sort(myVector.begin(),myVector.end(),std::greater<int>());
    Print("Sorted Standard Compare Function",myVector);
    // 3.以自定義的函數定義排序規則
    // lamda 表達式
    auto cmp1 = [](const int a,const int b)
    {
        return a < b;
    };
    std::sort(myVector.begin(),myVector.end(),cmp1);
    Print("Sorted Lamda Function",myVector);
    // 可調用對象
    struct
    {
        bool operator()(const int a,const int b){return a > b;}
    } cmp2;
    std::sort(myVector.begin(),myVector.end(),cmp2);
    Print("Sorted Function Object",myVector);
	
    return 0;
}

輸出結果為:

Sorted Default : 0 1 2 3 4 5 6 7 8 9
Sorted Standard Compare Function : 9 8 7 6 5 4 3 2 1 0
Sorted Lamda Function : 0 1 2 3 4 5 6 7 8 9
Sorted Function Object : 9 8 7 6 5 4 3 2 1 0

2. std::stable_sort(first,last)

使用std::sort()的一個問題是在排序時,無法保證值相等時的元素相對位置保持不變,如果程式中對相對順序有要求,那麼就需要使用std::stable_sort(),這是對std::sort()的一個升級版本,調用的方式和std::sort()相同,但是可以保證排序後的結果能夠保持原有的相對順序。

std::stable_sort()底層是基於歸併排序,時間複雜度是N * log(N)^2,如果可以使用額外的記憶體空間,那麼時間複雜度可以降低為N * log(N),std::sort()對容器和數組的要求與std::sort()相同。

不同庫對std::stable_sort()的實現:libstdc++libc++.

3. std::partial_sort(first,middle,last)

上述的std::sort()和std::stable_sort()都是對所選的範圍內的所有數據進行排序,但是如果對於一個容器或者數組,我們只需要找到其最小的幾個元素,那麼採用全局排序的方法效率將會很低,尤其是容器中的元素數量非常大時,將會對性能產生很大的影響。因此,C++標準庫提供了std::partial_sort()函數,來應用於這種場景。

std::partial_sort()函數的功能從其名字就可以得到,可以理解為部分排序,即從元素區間範圍內取出部分元素進行排序。函數參數列表中first,last表示元素範圍,middle用於定義需要進行排序的元素個數,具體的,std::partial_sort()會將範圍first-last範圍內最小(大)的middle-first個元素移動到first~middle範圍,並對這些元素進行升(降)序排序。

std::partial_sort()底層採用堆排序,時間複雜度為N * log(M),其中N為last-first,M為middle-first。由於底層實現的限制,std::partial_sort()對容器的要求與std::sort()和std::stable_sort()相同。

不同庫對std::stable_sort()的實現:libstdc++libc++.

示例代碼:

#include<iostream>
#include<vector>
#include<string>
#include<algorithm>
using namespace std;
void Print(std::string message,const std::vector<int>& vec)
{
    cout << message << ": ";
    for(const auto c : vec)
	cout << c << " ";
    cout <<endl;
}

int main()
{
    std::vector<int> myVector{5, 7, 4, 2, 8, 6, 1, 9, 0, 3};
    // 1. 以預設的排序規則
    std::partial_sort(myVector.begin(),myVector.begin() + 4,myVector.end());
    Print("Sorted Default", myVector);
    // 2. 以標準庫的greater函數進行排序
    std::partial_sort(myVector.begin(),myVector.end(),std::greater<int>());
    Print("Sorted Standard Compare Function",myVector);
    // 3.以自定義的函數定義排序規則
    // lamda 表達式
    auto cmp1 = [](const int a,const int b)
    {
        return a < b;
    };
    std::sort(myVector.begin(),myVector.end(),cmp1);
    Print("Sorted Lamda Function",myVector);
	
    return 0;
}

Sorted Default: 0 1 2 3 8 7 6 9 5 4
Sorted Standard Compare Function: 9 8 7 6 5 0 1 2 3 4
Sorted Lamda Function: 0 1 9 8 7 6 5 2 3 4

* C#中的Array.Sort()

筆者在用C++復刻C#代碼的時候發現對相同的數組排序的結果有時候會出現不一致,查了C#的官方文檔後才發現,C#中Array.Sort()函數是unstable_sort,也就是說其排序是無法保證相對順序的,而且Array似乎也沒有提供Stable_Sort版本的排序函數,因此如果需要保證相對順序不變,需要手動給原始的數據添加一個index,這樣再其他的key判等都相等時可以採用額外的Index來保持相對的原始序。而且有意思的是,Array.Sort()函數具體使用的排序方法是根據數據規模發生改變的

-如果數據量小於等於16個,則採用插入排序
-如果需要排序的數量超過2 * log(N),其中N為輸入的排序範圍,則採用堆排序
-其餘情況均採用快速排序
image

以上是對常用的標準庫排序演算法的總結和不同語言的對比,可以根據實際需要和數據量的大小來選擇合適的排序演算法。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 備忘錄模式(Memento Pattern):是一種行為型設計模式,在不破壞封裝性的前提下,捕獲一個對象的內部狀態,併在該對象之外保存這個狀態。在JavaScript中,可以使用閉包來實現備忘錄模式。 備忘錄模式通常用於處理用戶界面的狀態。當用戶與應用程式交互時,應用程式會根據用戶的輸入更改其狀態。 ...
  • 在基於vue-next-admin 的 Vue3+TypeScript 前端項目中,可以整合自己的 .NET 後端,前端操作一些功能的時候,為了使用方便全局掛載的對象介面,以便能夠快速處理一些特殊的操作,如消息提示、輔助函數、正則測試等等。本篇隨筆介紹在Vue3+TypeScript 前端項目中全局... ...
  • 眾所周知,網頁的暗黑模式可以減少屏幕反射和藍光輻射,減少眼睛的疲勞感,特別是在夜間使用時更為明顯。其實暗黑模式也給霓虹燈效應(Neon Effect)提供了發揮的環境。 霓虹燈效應是一種視覺效果,其特點是在深色背景上使用鮮艷的顏色來產生強烈的視覺衝擊。這種效應通常用於設計海報、廣告、標誌和網頁等。霓 ...
  • 領域驅動設計(Domain Driven Design,簡稱:DDD)設計思想和方法論早在2005年時候就被提出來,但是一直沒有被重視和推薦使用,直到2015年之後微服務流行之後,再次被人重視和推薦使用。 下麵我來介紹一下DDD設計思想和方法論,同時結合我們在實際項目中應用總結和思考。 目錄 1、為 ...
  • 一、案例背景 電腦包含記憶體(RAM),CPU 等硬體設備,根據如圖所示的“產品等級結構-產品族示意圖”,使用抽象工廠模式實現電腦設備創建過程並繪製類圖 二、實現步驟 根據題意,使用抽象工廠模式並畫出類圖,類圖中應包含一個抽象工廠類 AbstractFactory,PcFactory 和 MacF ...
  • 1. 理解可變性 1.1. 理解測試結果如何隨時間變化 1.2. 可以通過多次運行測試後取平均值來解決 1.3. 因代碼改進而進行的測試叫作回歸測試(regression testing) 1.3.1. 原本的代碼叫作基線(baseline) 1.3.2. 新的代碼叫作樣本(specimen) 1. ...
  • 用Python基於Google Bard做一個互動式的聊天機器人 之前已經通過瀏覽器試過了 Google Bard ,更多細節請看: Try out Google Bard, Will Google Bard beat the ChatGPT?. 現在我們想實現自動化,所以我用Python做一個交互 ...
  • 每次提交代碼的時候,你是否有為如何寫Commit Message而遲遲按不下提交的時刻呢?然後,死磨硬泡寫了一些並提交後,又被review的小伙伴吐槽了呢?相信很多小伙伴有過這樣的經歷吧? 趁著最近ChatGPT那麼火,就來順手推薦一個可以用於解決這個問題的VS Code插件:vscode-gpto ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...