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

統計

  • 隨筆 - 50
  • 文章 - 42
  • 評論 - 147
  • 引用 - 0

留言簿(6)

隨筆分類

文章分類

Link

搜索

  •  

積分與排名

  • 積分 - 167186
  • 排名 - 159

最新評論

閱讀排行榜

評論排行榜

STL中的排序算法一覽[By ACM郭老師]
這篇文章我很喜歡,是郭老師的新作!希望大家喜歡!
詳細的從算法的效率方面來說明了排序算法!

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

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

sort的基本使用方式如下:

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

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

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

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

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

sort_heap
堆排序也是一種快速的排序算法,復雜度也是O(N*logN)。STL中有一些和堆相關的函數,能夠構造堆,如果在構造好的堆上每次取出來根節點放在尾部,所有元素循環一遍,最后的結果也就有序了。這就是sort_heap了。它的使用要求區間前面已經構造成堆,如:
  1. C++:   
  2.   
  3. void func5()   
  4. {   
  5.     vector<int> ar;   
  6.     generate_n(back_inserter(ar), 100, rand);   
  7.     make_heap(ar.begin(), ar.end());   
  8.     sort_heap(ar.begin(), ar.end());   
  9. }    

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

其他
優先隊列(priority_queue)每次彈出的都是max值。實際上就是heap的一個容器方式的包裝。
關聯式容器自身就必須是有序的(針對key),對其迭代時,key是遞增的。
II、部分排序
這些部分排序功能能夠完成一段數據(而不是所有)的排序,在適當的適合使用可以節省計算量。不過用的人不多。

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

partial_sort接受三個參數,分別是區間的頭,中間和結尾。執行后,將前面M(M=中間-頭)個元素有序地放在前面,后面的元素肯定是比前面的大,但他們內部的次序沒有保證。partial_sort_copy的區別在于把結果放到另外指定的迭代器區間中:
  1. C++:   
  2.   
  3. void func6()   
  4. {   
  5.     int ar[12]={69,23,80,42,17,15,26,51,19,12,35,8};   
  6.     //只排序前7個數據   
  7.     partial_sort(ar, ar+7, ar+12);   
  8.     //結果是 8 12 15 17 19 23 26 80 69 51 42 35,后5個數據不定   
  9.     vector<int> res(7);   
  10.     //前7項排序后放入res   
  11.     partial_sort_copy(ar, ar+7, res.begin(), res.end(), greater<int>() );   
  12. }    

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

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

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

III、排序輔助功能
partition, stable_partition
merge, inplace_merge
IV、有序區間操作

這個準備單獨寫一篇

引用

讓我們繼續期待郭老師的補充!

郭老師的boke地址 www.skywind.name/blog

posted on 2009-05-25 11:51 pear_li 閱讀(498) 評論(0)  編輯 收藏 引用 所屬分類: C++

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            久久国产精品网站| 亚洲专区一区二区三区| 在线观看日产精品| 亚洲综合国产| 日韩性生活视频| 欧美成年人视频| 在线看一区二区| 美女999久久久精品视频| 性欧美videos另类喷潮| 国产伦精品一区二区三区视频黑人| 日韩午夜中文字幕| 亚洲精品少妇30p| 欧美激情影音先锋| 99riav1国产精品视频| 亚洲国产高清一区| 欧美极品在线视频| 亚洲视频一区在线| 在线观看日韩av电影| 久久一区二区三区国产精品 | 久久久亚洲精品一区二区三区| 国产欧美日韩亚洲精品| 欧美在线综合视频| 久久久精品999| 亚洲人成网站777色婷婷| 亚洲欧洲一区| 国产精品久久久久久久久久三级| 欧美亚洲一区二区在线观看| 亚洲影音先锋| 国产一区二区三区日韩欧美| 蜜臀av性久久久久蜜臀aⅴ| 美女国内精品自产拍在线播放| 91久久在线| 亚洲视频网站在线观看| 国产在线精品一区二区夜色| 欧美11—12娇小xxxx| 欧美人与禽性xxxxx杂性| 亚洲性感激情| 欧美一区国产二区| 亚洲精品美女| 亚洲综合色在线| 亚洲国产精品久久久久秋霞不卡| 亚洲精品少妇| 国产综合精品一区| 亚洲精品久久久久久下一站| 国产精品日韩欧美| 欧美福利电影在线观看| 国产精品护士白丝一区av| 久久久久成人精品| 欧美日韩直播| 欧美大片免费久久精品三p| 欧美日韩国产影院| 久久精品亚洲国产奇米99| 你懂的国产精品| 欧美在线电影| 欧美日韩国产成人在线观看| 久久久久久久波多野高潮日日 | 99热精品在线| 精品粉嫩aⅴ一区二区三区四区| 日韩视频免费在线观看| 欧美gay视频| 西西人体一区二区| 免费在线亚洲欧美| 久久嫩草精品久久久久| 国产精品白丝av嫩草影院| 免费亚洲电影| 国产午夜一区二区三区| 夜夜狂射影院欧美极品| 在线欧美不卡| 欧美与黑人午夜性猛交久久久| 亚洲一区二区三区在线播放| 老司机成人网| 国产一区二区高清| 一本大道久久a久久精品综合| 在线精品视频一区二区| 欧美亚洲一区二区在线| 亚洲伊人伊色伊影伊综合网| 欧美激情国产日韩| 欧美激情91| 一区二区在线视频观看| 午夜在线观看欧美| 久久精品国产综合| 国产精品一区二区三区久久久 | 99在线|亚洲一区二区| 亚洲国产精品久久久久秋霞不卡 | 欧美精品成人一区二区在线观看 | 亚洲特色特黄| 亚洲午夜精品国产| 欧美日韩一区二区三区四区五区| 欧美+亚洲+精品+三区| 曰韩精品一区二区| 久久久99精品免费观看不卡| 久久久青草青青国产亚洲免观| 国产精品毛片a∨一区二区三区|国| 亚洲精品在线电影| 在线亚洲一区二区| 欧美亚日韩国产aⅴ精品中极品| 在线视频欧美日韩| 午夜精品久久久久| 国产日韩欧美综合| 久久久成人精品| 欧美高潮视频| 在线视频日韩| 国产精品一区二区黑丝| 久久aⅴ乱码一区二区三区| 久久久人成影片一区二区三区| 国产综合色产在线精品| 媚黑女一区二区| 日韩系列欧美系列| 久久国产精品99久久久久久老狼 | 久久久国产午夜精品| 在线成人激情| 欧美日韩另类国产亚洲欧美一级| 9色精品在线| 欧美与欧洲交xxxx免费观看 | 一本色道久久88亚洲综合88| 欧美视频久久| 麻豆freexxxx性91精品| 亚洲国产一区二区三区a毛片| 欧美二区不卡| 亚洲资源av| 国内精品久久久久影院 日本资源 国内精品久久久久伊人av | 亚洲国产成人久久| 亚洲欧美日韩精品在线| 国产一区二区丝袜高跟鞋图片| 老司机一区二区三区| 亚洲免费av观看| 久久国产欧美精品| 日韩视频三区| 国产一区二区欧美| 欧美日韩国产影片| 久久精品欧洲| 亚洲免费精品| 猛男gaygay欧美视频| 亚洲愉拍自拍另类高清精品| 在线观看精品视频| 国产欧美日韩另类一区| 欧美激情片在线观看| 亚洲欧美日韩久久精品| 91久久线看在观草草青青| 久久精品亚洲国产奇米99| 一区二区三区免费看| 亚洲成色777777女色窝| 国产精品自拍视频| 欧美日韩精品在线播放| 老司机午夜精品视频| 亚洲欧美日韩中文在线制服| 亚洲国产另类久久久精品极度| 久久国产高清| 亚洲欧美中文另类| 一区二区成人精品 | 欧美在线视频二区| 夜夜嗨av一区二区三区四季av| 好看不卡的中文字幕| 国产精品多人| 欧美日韩免费观看一区=区三区| 老司机午夜免费精品视频 | 狠狠色狠色综合曰曰| 国产精品v欧美精品v日韩精品| 欧美大片在线观看一区二区| 欧美在线精品免播放器视频| 亚洲一品av免费观看| 亚洲精品视频一区| 亚洲激情视频网| 欧美国产免费| 蜜臀av在线播放一区二区三区| 欧美一区二区福利在线| 亚洲综合色自拍一区| 亚洲一区二区三区免费视频| 一本不卡影院| 一本大道久久a久久综合婷婷| 亚洲日本免费| 亚洲精品国产欧美| 亚洲日本va在线观看| 亚洲人成高清| 999亚洲国产精| 亚洲免费观看高清完整版在线观看熊| 最新亚洲视频| 一本综合精品| 中文成人激情娱乐网| 亚洲视频你懂的| 亚洲欧美三级伦理| 久久精品国产精品亚洲综合| 国产精品一区久久久久| 国产精品麻豆va在线播放 | 欧美大片专区| 亚洲欧洲美洲综合色网| 午夜精品国产精品大乳美女| 亚洲男女毛片无遮挡| 亚洲欧美综合v| 欧美影视一区| 女生裸体视频一区二区三区| 亚洲高清一区二| 99精品国产在热久久婷婷| 亚洲视频axxx| 久久精品二区| 免费亚洲一区二区| 国产精品成人在线| 国内揄拍国内精品少妇国语| 亚洲日本无吗高清不卡| 亚洲午夜极品|