青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

隨筆 - 8  文章 - 26  trackbacks - 0
<2025年9月>
31123456
78910111213
14151617181920
21222324252627
2829301234
567891011

常用鏈接

留言簿(5)

隨筆檔案

文章分類

文章檔案

相冊

C++語言

搜索

  •  

最新評論

閱讀排行榜

評論排行榜

//By skywind

STL中有多種排序算法,各有各的適用范圍,下面聽我一一道來:

I、完全排序

  1. sort()

首先要隆重推出的當然是最最常用的sort了,sort有兩種形式,第一種形式有兩個迭代器參數,構成一個前開后閉的區間,按照元素的 less 關系排序;第二種形式多加一個指定排序準則的謂詞。sort基本是最通用的排序函數,它使用快速排序算法,并且在遞歸過程中,當元素數目小于一個閾值(一般是16,我的試驗是24)時,轉成直接插入排序。偉大的數學家Knuth已經證明,在平均意義上,快速排序是最快的了;當然,最壞復雜性比較差。sort要求隨機迭代器,因此對于很多編譯器來說,對于前向迭代器(如list)使用sort是一個編譯錯誤。(不過,在vc2005里面,這個錯誤信息實在很糟糕)

sort的基本使用方式如下:

PLAIN TEXT

C++:

    1. #include <vector>
    2. #include <algorithm>
    3. #include <functional>
    4. #include <cstdlib>
    5.  
    6. using namespace std;
    7.  
    8. void func1()
    9. {
    10.     vector<int> ar;
    11.     //向數組里面插入一些隨機數
    12.     generate_n(back_inserter(ar), 100, rand);
    13.     //按從小到大排序
    14.     sort(ar.begin(), ar.end());
    15. }

經常有人問如何從大到小逆排序,這個其實有很多種方式實現,如下面的例子:

PLAIN TEXT

C++:

    1. void func2()
    2. {
    3.     vector<int> ar;
    4.     //向數組里面插入一些隨機數
    5.     generate_n(back_inserter(ar), 100, rand);
    6.  
    7.     //方法1:使用函數作為謂詞
    8.     sort(ar.begin(), ar.end(), GreateThan);
    9.     //方法2:使用仿函數作為謂詞
    10.     //注意下面兩種方法都需要有個括號,實際上是要產生一個臨時對象
    11.     sort(ar.begin(), ar.end(), CompareInt());
    12.     //方法3:使用預定義的Adapter, 定義在 <functional>
    13.     sort(ar.begin(), ar.end(), greater<int>());
    14.     //方法4:正常排序,然后翻轉過來
    15.     sort(ar.begin(), ar.end());
    16.     reverse(ar.begin(), ar.end());
    17.     //方法5:使用逆迭代器
    18.     sort(ar.rbegin(), ar.rend());
    19. }

最后一種方法是我比較欣賞的,可以不能直接對原生數組使用,也就是說,如果ar的定義是int ar[MAXN],上面其他的排序算法都可以簡單的改成sort(ar, ar+MAXN, ...),但最后一個不行,要用另外一種比較丑陋的方式:

PLAIN TEXT

C++:

    1. #include <iterator>
    2. void func3(){
    3.     int ax[5]={1,3,4,5,2};
    4.     sort(reverse_iterator<int*>(ax+5), reverse_iterator<int*>(ax+0));
    5. }
  1. stable_sort

sort優點一大堆,一個缺點就是它不是一種穩定的排序。什么是排序的穩定性?就是如果出現兩個元素相等時,要求排序之后他們之間保持原來的次序(比如我們先按學號排序,然后按成績排序,這時就希望成績相同的還是按照學號的次序排)。很可惜,快速排序算法就不是穩定的,要追求這個,只好用stable_sort了。

在各種排序算法中,合并排序是穩定的,但一般的合并排序需要額外的O(N)的存儲空間,而這個條件不是一定能夠滿足的(可能是比較奢侈的)。所以在stable_sort內部,首先判斷是否有足夠的額外空間(如vecotr中的cap()-size()部分),有的話就使用普通合并函數,總的時間復雜性和快速排序一個數量級,都是O(N*logN)。如果沒有額外空間,使用了一個merge_without_buffer的關鍵函數進行就地合并(如何實現是比較有技巧的,完全可以專門談一談),這個合并過程不需要額外的存儲空間(遞歸的堆棧除外),但時間復雜度變成O(N*logN),這種情況下,總的stable_sort時間復雜度是O(N*logN*logN)

總之,stable_sort稍微慢一點兒,但能夠保證穩定,使用方法和sort一樣。但很多時候可以不用這種方式和這個函數,比如上面的例子,完全可以在排序比較準則中寫入成績和學號兩個條件就OK了。

PLAIN TEXT

C++:

    1. class CStudent
    2. {
    3. public:
    4.     CStudent();
    5.     //注意這個比較函數中的const
    6.     bool operator<(const CStudent& rhs) const
    7.     {
    8.         if (m_score != rhs.m_score)
    9.             return (m_score <rhs.m_score);
    10.         return m_name <rhs.m_name;
    11.     }
    12. protected:
    13.     std::string m_name;
    14.     int m_score;
    15. };
    16.  
    17. void func4()
    18. {
    19.     vector<CStudent> arStu;
    20.     sort(arStu.begin(), arStu.end());
    21. }
  1. sort_heap

堆排序也是一種快速的排序算法,復雜度也是O(N*logN)STL中有一些和堆相關的函數,能夠構造堆,如果在構造好的堆上每次取出來根節點放在尾部,所有元素循環一遍,最后的結果也就有序了。這就是sort_heap了。它的使用要求區間已經被構造成堆,如:

PLAIN TEXT

C++:

    1. void func5()
    2. {
    3.     vector<int> ar;
    4.     //生成數據
    5.     generate_n(back_inserter(ar), 100, rand);
    6.     //構造堆
    7.     make_heap(ar.begin(), ar.end());
    8.     //堆排序
    9.     sort_heap(ar.begin(), ar.end());
    10. }
  1. list.sort

對于list容器,是不能直接使用sort的(包括stable_sort),從技術的角度來說,sort要求隨機迭代器;從算法的角度來說,list這種鏈表結構本身就不適合用快速排序。因此,list容器內部實現了專門的sort算法,這個算法采用的是合并排序,應該是穩定的(不確定)。如:

PLAIN TEXT

C++:

    1. list<int> li;
    2.     li.sort();
  1. 其他
    • 優先隊列(priority_queue)每次彈出的都是max值。實際上就是heap的一個容器方式的包裝。
    • 關聯式容器自身就必須是有序的(針對key),對其迭代時,key是遞增的。

II、部分排序

這些部分排序功能能夠完成一段數據(而不是所有)的排序,在適當的適合使用可以節省計算量。用的人不多,可能了解的也比較少。

  1. partial_sort(), partial_sort_copy()

這兩個函數能夠將整個區間中給定數目的元素進行排序,也就是說,結果中只有最小的M個元素是有序的。你當然也可以使用sort,區別就在于效率。如果M顯著地小于N,時間就比較短;當然M太小了也不好,那還不如挨個找最小值了。

partial_sort接受三個參數,分別是區間的頭,中間和結尾。執行后,將前面MM=中間-頭)個元素有序地放在前面,后面的元素肯定是比前面的大,但他們內部的次序沒有保證。partial_sort_copy的區別在于把結果放到另外指定的迭代器區間中:

PLAIN TEXT

C++:

    1. void func6()
    2. {
    3.     int ar[12]={69,23,80,42,17,15,26,51,19,12,35,8};
    4.     //只排序前7個數據
    5.     partial_sort(ar, ar+7, ar+12);
    6.     //結果是 8 12 15 17 19 23 26 80 69 51 42 35,后5個數據次序不定,依賴于實現
    7.     vector<int> res(7);
    8.     //7項排序后放入res
    9.     partial_sort_copy(ar, ar+7, res.begin(), res.end(), greater<int>() );
    10. }

這兩個函數的實現使用的是堆的方法,先將前M個元素構造成堆,然后挨個檢查后面的元素,看看是否小于堆的最大值,是的話就彼此交換,然后重排堆;最后將前面已經是最小的M個元素構成的堆作一次sort_heap就可以了。算法的復雜度差不多是O(N*logM)

  1. nth_element

這個函數只真正排序出一個元素來,就是第n個。函數有三個迭代器的輸入(當然還可以加上一個謂詞),執行完畢后,中間位置指向的元素保證和完全排序后這個位置的元素一致,前面區間的元素都小于(精確地說,是不大于)后面區間的元素。

熟悉快速排序的馬上就能發現,這實際上是一個按位置劃分的算法。STL的規范中要求此函數的平均復雜度是線性的,和快速排序一樣,這種算法的最壞復雜度比較差。在一般的實現(如SGI)中,采用三種取1的方法尋找劃分元素,最壞復雜度是O(N^N)。雖然理論上有一些算法可以保證最壞線性復雜度,但算法過于復雜,STL一般也不采用。

III、排序輔助功能

  1. partition, stable_partition
  2. merge, inplace_merge

IV、有序區間操作

這個準備單獨寫一篇

 

posted on 2008-06-23 20:25 楊彬彬 閱讀(6824) 評論(0)  編輯 收藏 引用 所屬分類: STL
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            另类天堂av| 欧美一区二区视频在线| 亚洲欧美日韩精品一区二区| 亚洲激情一区| 亚洲国内在线| 亚洲精品免费看| 亚洲欧洲在线免费| 亚洲精品一区久久久久久| 亚洲三级视频在线观看| 亚洲免费大片| 国产精品v欧美精品v日本精品动漫| 欧美精品系列| 欧美涩涩网站| 国产精品无人区| 精品动漫3d一区二区三区免费版| 黄色在线一区| 日韩视频在线免费观看| 亚洲一二三区视频在线观看| 午夜免费日韩视频| 久久一区中文字幕| 亚洲日本va在线观看| 欧美激情亚洲自拍| 宅男在线国产精品| 久久久久久噜噜噜久久久精品| 欧美大片国产精品| 国产精品入口尤物| 亚洲国产精品成人综合色在线婷婷| 99热这里只有成人精品国产| 欧美在线视频一区| 欧美国产第二页| 亚洲香蕉成视频在线观看 | 午夜精品久久久99热福利| 久久国产日韩| 欧美日韩另类丝袜其他| 黄色av日韩| 亚洲欧美在线一区二区| 欧美大片在线观看| 亚洲女同精品视频| 免费黄网站欧美| 国产亚洲成av人在线观看导航| 亚洲人午夜精品| 久久久久久久欧美精品| 一区二区三区精密机械公司| 噜噜噜在线观看免费视频日韩| 国产精品乱码| 99国产精品久久久久久久久久 | 国产精品美女久久久久久2018| 国产午夜精品久久久久久免费视| 亚洲久色影视| 久久久国产视频91| 在线视频日本亚洲性| 欧美激情精品久久久久久大尺度 | 中文一区在线| 欧美激情影音先锋| 亚洲第一伊人| 久久综合99re88久久爱| 亚洲欧美激情四射在线日 | 蜜桃av久久久亚洲精品| 国内精品视频一区| 久久精品国语| 久久国产精品久久w女人spa| 艳妇臀荡乳欲伦亚洲一区| 欧美激情一区二区三区成人| 91久久精品一区二区三区| 免费观看成人鲁鲁鲁鲁鲁视频| 午夜精品久久久久| 国产区亚洲区欧美区| 午夜老司机精品| 亚洲欧美成人一区二区在线电影| 国产精品国产三级欧美二区| 亚洲性夜色噜噜噜7777| 中文精品视频| 国产乱码精品1区2区3区| 欧美一级大片在线免费观看| 亚洲无限av看| 国产人妖伪娘一区91| 久久精品午夜| 玖玖国产精品视频| 亚洲久久成人| 一区二区三区国产在线观看| 国产精品美女久久| 久久精品国产第一区二区三区最新章节| 亚洲自拍高清| 激情91久久| 亚洲国产欧美一区二区三区久久| 欧美韩日亚洲| 午夜电影亚洲| 久久久久青草大香线综合精品| 亚洲国产精品日韩| 亚洲另类黄色| 国产日韩在线播放| 欧美国产日本在线| 国产精品大全| 免费不卡在线观看av| 欧美日韩国产综合一区二区| 性欧美videos另类喷潮| 久久久久久久一区二区| 国产精品久久久久久超碰| 欧美在线一二三区| 久久精品视频免费播放| 亚洲剧情一区二区| 欧美一区激情| 亚洲精品视频在线观看网站| 午夜精品久久久久久久久 | 久久成人免费网| 欧美国产91| 久久在线免费观看| 欧美日韩蜜桃| 欧美大香线蕉线伊人久久国产精品| 国产精品爽爽ⅴa在线观看| 亚洲二区精品| 黑丝一区二区三区| 亚洲一区二区三区高清不卡| 亚洲三级免费| 久久久蜜桃一区二区人| 欧美在线免费| 欧美午夜免费电影| 亚洲精品中文字幕在线观看| 黄色一区二区在线| 午夜在线一区二区| 日韩亚洲在线| 免费看黄裸体一级大秀欧美| 久久久久久久久久看片| 国产精品久久久久久久久搜平片| 久久综合影音| 国产欧美一区二区色老头| 在线视频中文亚洲| 亚洲日本aⅴ片在线观看香蕉| 99国产精品视频免费观看| 亚洲伦理在线免费看| 欧美在线网址| 亚洲欧美视频一区| 中文精品在线| 亚洲欧美日韩爽爽影院| 欧美韩日一区| 欧美成人一区二区三区片免费| 国产精品日本| 亚洲欧美日韩国产中文在线| 在线午夜精品| 欧美金8天国| 欧美激情中文字幕乱码免费| 狠狠做深爱婷婷久久综合一区| 亚洲性图久久| 一本色道久久综合亚洲精品婷婷| 久久久天天操| 一级成人国产| 欧美亚州在线观看| 亚洲激情自拍| 亚洲精品乱码久久久久久久久| 亚洲欧美不卡| 91久久精品美女高潮| 欧美精品九九99久久| 亚洲第一级黄色片| 在线播放中文字幕一区| 欧美在线视频网站| 亚洲一区二区综合| 欧美日韩一区在线| av不卡免费看| 亚洲一区欧美二区| 欧美日韩喷水| 一区二区欧美精品| 亚洲国产精品黑人久久久| 欧美精品一区二区高清在线观看| 亚洲精品久久视频| 麻豆国产va免费精品高清在线| 久久嫩草精品久久久精品一| 亚洲高清123| 欧美激情1区| 99精品视频一区二区三区| 夜夜嗨av一区二区三区中文字幕 | 欧美激情视频一区二区三区免费| 欧美黄色日本| 亚洲日韩视频| 久久久久久久精| 99精品久久免费看蜜臀剧情介绍| 亚洲一本视频| 国产日韩成人精品| 久久久久天天天天| 亚洲影院色无极综合| 久久久精彩视频| 亚洲国产成人porn| 欧美日韩亚洲不卡| 亚洲电影网站| 久久综合给合久久狠狠色 | 久久精品国产精品亚洲综合 | 亚洲免费观看高清在线观看| 亚洲午夜伦理| 在线日韩成人| 欧美日韩国产色综合一二三四| 亚洲美女免费精品视频在线观看| 亚洲综合成人在线| 一区二区三区亚洲| 欧美精品v日韩精品v国产精品| 中日韩午夜理伦电影免费| 欧美专区福利在线| 亚洲激情在线| 欧美粗暴jizz性欧美20| 亚洲五月六月| 亚洲福利视频一区二区| 欧美亚洲视频|