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

隨筆-4  評論-40  文章-117  trackbacks-0


這篇文章我很喜歡,是郭老師的新作!希望大家喜歡!
詳細(xì)的從算法的效率方面來說明了排序算法!

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

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

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.     //向數(shù)組里面插入一些隨機(jī)數(shù)   
  14.     generate_n(back_inserter(ar), 100, rand);   
  15.     //按從小到大排序   
  16.     sort(ar.begin(), ar.end());   
  17. }    

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

最后一種方法是我比較欣賞的,可以不能直接對原生數(shù)組使用,也就是說,如果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優(yōu)點(diǎn)一大堆,一個缺點(diǎn)就是它不是一種穩(wěn)定的排序。什么是排序的穩(wěn)定性,就是如果出現(xiàn)兩個元素相等時,要求排序之后他們之間保持原來的次序(比如我們先按學(xué)號排序,然后按成績排序,這時就希望成績相同的還是按照學(xué)號的次序排)。很可惜,快速排序算法就不是穩(wěn)定的,要追求這個,只好用stable_sort了。

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

總之,stable_sort稍微慢一點(diǎn)兒,但能夠保證穩(wěn)定,使用方法和sort一樣。但很多時候可以不用這種方式和這個函數(shù),比如上面的例子,完全可以在排序比較準(zhǔn)則中寫入成績和學(xué)號兩個條件就OK了
  1. C++:   
  2.   
  3. class CStudent   
  4. {   
  5. public:   
  6.     CStudent();   
  7.     //注意這個比較函數(shù)中的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
堆排序也是一種快速的排序算法,復(fù)雜度也是O(N*logN)。STL中有一些和堆相關(guān)的函數(shù),能夠構(gòu)造堆,如果在構(gòu)造好的堆上每次取出來根節(jié)點(diǎn)放在尾部,所有元素循環(huán)一遍,最后的結(jié)果也就有序了。這就是sort_heap了。它的使用要求區(qū)間前面已經(jīng)構(gòu)造成堆,如:
  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),從技術(shù)的角度來說是由于sort要求隨機(jī)迭代器;從算法的角度來說,list這種鏈表結(jié)構(gòu)就不適合用快速排序。因此,list容器內(nèi)部實(shí)現(xiàn)了專門的sort算法,這個算法采用的是合并排序,應(yīng)該是穩(wěn)定的(不確定)。

其他
優(yōu)先隊(duì)列(priority_queue)每次彈出的都是max值。實(shí)際上就是heap的一個容器方式的包裝。
關(guān)聯(lián)式容器自身就必須是有序的(針對key),對其迭代時,key是遞增的。
II、部分排序
這些部分排序功能能夠完成一段數(shù)據(jù)(而不是所有)的排序,在適當(dāng)?shù)倪m合使用可以節(jié)省計(jì)算量。不過用的人不多。

partial_sort(), partial_sort_copy()
這兩個函數(shù)能夠?qū)⒄麄€區(qū)間中給定數(shù)目的元素進(jìn)行排序,也就是說,結(jié)果中只有最小的M個元素是有序的。你當(dāng)然也可以使用sort,區(qū)別就在于效率。如果M顯著地小于N,時間就比較短;當(dāng)然M太小了也不好,那還不如挨個找最小值了。

partial_sort接受三個參數(shù),分別是區(qū)間的頭,中間和結(jié)尾。執(zhí)行后,將前面M(M=中間-頭)個元素有序地放在前面,后面的元素肯定是比前面的大,但他們內(nèi)部的次序沒有保證。partial_sort_copy的區(qū)別在于把結(jié)果放到另外指定的迭代器區(qū)間中:
  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個數(shù)據(jù)   
  7.     partial_sort(ar, ar+7, ar+12);   
  8.     //結(jié)果是 8 12 15 17 19 23 26 80 69 51 42 35,后5個數(shù)據(jù)不定   
  9.     vector<int> res(7);   
  10.     //前7項(xiàng)排序后放入res   
  11.     partial_sort_copy(ar, ar+7, res.begin(), res.end(), greater<int>() );   
  12. }    

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

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

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

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

這個準(zhǔn)備單獨(dú)寫一篇

引用

讓我們繼續(xù)期待郭老師的補(bǔ)充!

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



posted on 2009-03-12 15:55 李陽 閱讀(569) 評論(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>
            亚洲激情国产| 久久黄色小说| 免费亚洲婷婷| 久久成人综合视频| 国模 一区 二区 三区| 久久亚洲综合色一区二区三区| 亚洲深爱激情| 国产日韩欧美在线观看| 久久久久久9999| 另类专区欧美制服同性| 亚洲毛片一区| 亚洲欧美精品伊人久久| 1769国产精品| 亚洲性感激情| 亚洲国语精品自产拍在线观看| 亚洲国产精品综合| 欧美日韩国产精品自在自线| 欧美一区二区成人6969| 欧美高清在线视频| 久久精品视频99| 欧美日韩成人网| 欧美va亚洲va日韩∨a综合色| 欧美激情一区二区三区蜜桃视频| 欧美一区二区三区在线观看视频| 另类人畜视频在线| 免费成人激情视频| 欧美与黑人午夜性猛交久久久| 性欧美大战久久久久久久久| 亚洲精品资源| 欧美福利一区| 欧美成人一二三| 国产一区亚洲一区| 欧美一区二区三区播放老司机| 亚洲看片一区| 欧美精品18| 亚洲人精品午夜在线观看| 亚洲国产91精品在线观看| 久久天堂av综合合色| 欧美成人综合一区| 亚洲精品乱码久久久久久久久 | 欧美一区二区三区成人| 欧美激情欧美狂野欧美精品| 欧美wwwwww| 一本色道久久综合| 欧美高清视频免费观看| 亚洲黄色毛片| 午夜精品福利电影| 国内精品国产成人| 欧美aa在线视频| 亚洲私拍自拍| 欧美成人dvd在线视频| 亚洲免费黄色| 国产一级久久| 欧美日韩国产综合一区二区| 中国日韩欧美久久久久久久久| 欧美伊人精品成人久久综合97| 国产免费亚洲高清| 欧美高清你懂得| 亚洲欧美色婷婷| 亚洲精品少妇30p| 久久人人爽爽爽人久久久| 亚洲精品资源| 亚洲国产另类精品专区 | 久久婷婷成人综合色| 亚洲久久一区| 今天的高清视频免费播放成人| 欧美区一区二| 你懂的视频一区二区| 香蕉国产精品偷在线观看不卡| 亚洲二区视频在线| 美女视频黄a大片欧美| 欧美在线|欧美| 亚洲国产福利在线| 国产字幕视频一区二区| 欧美午夜精品久久久久久浪潮| 老鸭窝亚洲一区二区三区| 久久久夜夜夜| 蘑菇福利视频一区播放| 麻豆精品精华液| 欧美激情网站在线观看| 欧美成人日本| 国产精品久久久久9999吃药| 欧美日韩国产片| 国产精品wwwwww| 国产精品女人毛片| 国产欧美精品xxxx另类| 国产亚洲成精品久久| 国内伊人久久久久久网站视频| 国产精品亚洲а∨天堂免在线| 欧美性猛交xxxx乱大交蜜桃| 国产精品黄页免费高清在线观看| 国产精品视频不卡| 国产一区二区av| 99成人在线| 久久精品亚洲热| 亚洲电影在线看| 羞羞视频在线观看欧美| 久久夜色精品亚洲噜噜国产mv| 欧美激情小视频| 国产在线国偷精品产拍免费yy| 亚洲国产日韩欧美在线99 | 亚洲毛片视频| 亚洲精品资源| 亚洲人成人一区二区三区| 亚洲精品一区二区三区樱花 | 亚洲乱码一区二区| 久久精品国产免费| 国产精品一区二区在线观看不卡| 精品成人a区在线观看| 午夜在线观看免费一区| 亚洲欧洲一区二区三区久久| 久久久91精品国产一区二区精品| 欧美视频中文一区二区三区在线观看| 精品1区2区3区4区| 久久国产欧美日韩精品| 在线一区欧美| 欧美精品在线免费| 在线一区欧美| 日韩亚洲一区在线播放| 欧美国产亚洲精品久久久8v| 在线观看视频一区二区欧美日韩| 亚洲欧美久久久久一区二区三区| 亚洲精品国产视频| 欧美日本免费| 亚洲一区精品在线| 亚洲在线视频网站| 国产一区二区三区四区在线观看 | 国产一区二区三区免费观看| 欧美一区二区私人影院日本 | 亚洲久久一区二区| 亚洲天堂av高清| 国产在线精品二区| 亚洲丰满在线| 国产精品久久久久久久一区探花| 欧美亚洲自偷自偷| 久久中文欧美| 亚洲免费在线观看| 久久久久网站| 一本色道久久综合狠狠躁的推荐| 亚洲欧洲在线视频| 久久aⅴ国产紧身牛仔裤| 亚洲人成欧美中文字幕| 亚洲一区二区不卡免费| 亚洲视频在线二区| 亚洲高清成人| 欧美+日本+国产+在线a∨观看| 久久九九免费视频| 欧美日韩在线视频首页| 99爱精品视频| 亚洲日韩欧美一区二区在线| 欧美一级欧美一级在线播放| 欧美精品一卡二卡| 国产精品久久国产三级国电话系列| 国产精品裸体一区二区三区| 在线日韩一区二区| 欧美一区二区三区的| 欧美亚洲网站| 国产精品手机视频| 亚洲欧美日韩在线一区| 久久人人超碰| 久久手机免费观看| 国产一区二区三区视频在线观看| 亚洲黄一区二区三区| 在线日韩av| 欧美成人综合一区| 亚洲国产精品va在看黑人| 1024成人| 欧美日韩福利视频| 亚洲精品网址在线观看| 99国产精品久久久久老师 | 亚洲网址在线| 亚洲欧洲综合| 激情欧美一区二区三区| 亚洲第一精品夜夜躁人人躁 | 欧美xxx成人| 国产主播精品在线| 米奇777在线欧美播放| 亚洲国产欧洲综合997久久| 一本色道久久88综合亚洲精品ⅰ | 亚洲国产精品一区二区www| 欧美激情1区2区3区| 亚洲免费观看| 乱中年女人伦av一区二区| 亚洲欧洲一区二区天堂久久| 欧美日韩视频一区二区| 欧美自拍丝袜亚洲| 亚洲视频欧美在线| 久久久精彩视频| 这里只有精品视频在线| 禁久久精品乱码| 国产精品一区毛片| 欧美国产欧美综合 | 亚洲视频一二区| 亚洲日本欧美在线| 好看不卡的中文字幕| 欧美三级在线播放| 欧美激情一区二区三级高清视频| 欧美一区久久| 午夜亚洲福利在线老司机| 亚洲视频一起|