• <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>

            雁過無痕

              C++博客 :: 首頁 :: 新隨筆 :: 聯(lián)系 :: 聚合  :: 管理 ::

            《編程之美》讀書筆記072.5 尋找最大的K個數

             

            問題:

            n個數中找出最大的k個數。

             

            兩種思路:

            1 保存目前找到的最大k個數,每訪問一個數,就與這k個數中的最小值比較,決定是否更新這k個數。儲存k個數的數據結構可采用:敗者樹、二叉查找樹、最小堆。

            C++ STL提供了multisetpriority_queue容器,另外還提供了make_heappush_heappop_heap方便手動構建堆結構。(測試發(fā)現(xiàn),手工建堆的效率最高,當nk增大到一定值時,采用紅黑樹的multiset的效率極差。手動建堆的效率相比priority_queue有略微提高。)

             

            2 修改排序方法,去除不必要的過程。

            選擇排序: 只要選k次。

            冒泡排序: 只要冒泡k次即可。

            堆排序:   構建好最大堆后,取 k次最大值

            快速排序: 分區(qū)時,根據數P將數組分為兩部分,設大于P的數個數為a,小于P的數的個數為b。如果,a>=k,則從這a個數取最大的k個數,若a<k,則從b個數取最大的k-a-1個。

            歸并排序: 當待合并的兩個數組,兩數組長度和大等于k時,合并時只取前k個。或者:以(k+1)/2個數為一組,將數組分成幾個組,對每組進行排序(可以采用任何一種高效的排序方法)后,兩兩合并時只取前k個。

            計數排序: 如果都是整數,先掃描一遍找出最大值max,最小值min,再掃一遍,將每個值減去min,對這個值計數,最后從max-min開始統(tǒng)計,找出最大的k個數。另外,也可采用桶排序。

            桶排序:   可以不對桶內的數據進行排序。

            基數排序: 可以采用最高關鍵字比較方法,并免去相關的排序。

             

            STL中的nth_element就是基于對intorsort的修改(introtsort是對快速排序的改進,當遞歸深度達到一定值時,可切換到堆排序),而partial_sortpartial_sort_copy是基于堆排序的修改,因而在k很小時,其效率可能會高于nth_element。遺憾的是:STL沒有提供完全基于堆排序的nth_element

             

            從下面的測試結果,可以看出:在M不是很大,M/N很小時,partial_sortpartial_sort_copy盡管多了“對堆結構進行排序”這個不必要的操作,其效率仍然高于nth_element,但相差不多。而在其它情況下nth_element的效率則比其它的幾種方法要高很多。

             

            如果源數據都是整數,多數情況下(即使允許修改源數據),桶排序方法(結合計數方法)的效率比nth_element高。桶排序只需256K的內存,效率很高。在MN至少有一個大于當前內存大小的情況下,桶排序是最佳選擇,其性能遠高于其它方法。

             

            如果源數據是浮點數,根據浮點數在內存中的表示,可以對桶排序方法進行適當修改,使之對浮點數也適用。

             

            測試程序相關說明:

            ① 測試程序要求不得改變源數據,某些方法要多一個復制源數據操作,可以從partial_sort_copypartial_sort效率的差異,看出這個復制操作的影響。

            ② 桶排序方法對應nth_count

            ③ 對堆結構的調整,采用三種途徑(分別對應三個程序):利用push_heappop_heap、只用pop_heap、手寫代碼調整。(

            multisetheapsort方法,在相同NM情況下,所用時間起伏很大,即所用時間對原始數據依賴性很高。

             

            程序代碼

            posted on 2010-08-16 00:02 flyinghearts 閱讀(2807) 評論(1)  編輯 收藏 引用 所屬分類: 編程之美

            評論

            # re: 《編程之美》讀書筆記07: 2.5 尋找最大的K個數 2014-08-12 13:59 121e1212
            int test[][2] = {{5, 5}, {5, 7}, {7, 5}, {4, 4}, {4, 6}, {6, 4}};

            const int sz = sizeof(test) / sizeof(test[0]);

            std::cout << "Test 2:\n";http://www.ssnz88.net

            for (int i = 0; i < sz; ++i) {

            int row = test[i][0];

            int col = test[i][1];

            int **arr = new int*[row];

            for (int i = 0; i < row; ++i) arr[i] = new int[col];
              回復  更多評論
              

            久久91精品国产91久久户| 精品少妇人妻av无码久久| 日韩va亚洲va欧美va久久| 亚洲国产美女精品久久久久∴ | 9999国产精品欧美久久久久久| 国产精品久久久久久福利69堂| 久久久久无码精品国产app| 亚洲AV无一区二区三区久久| 久久久久无码专区亚洲av| 久久国产V一级毛多内射| 久久精品国产亚洲av水果派| 国产精品久久久久无码av| 精品乱码久久久久久夜夜嗨| 免费一级做a爰片久久毛片潮| 久久天天躁狠狠躁夜夜不卡| 久久久久人妻一区二区三区vr| 一级做a爱片久久毛片| 久久这里只有精品视频99| 综合久久精品色| 精品视频久久久久| 一本色道久久综合狠狠躁| 久久免费视频一区| 久久精品中文闷骚内射| 久久精品夜夜夜夜夜久久| 99精品国产免费久久久久久下载 | 狠狠综合久久综合中文88| 奇米影视7777久久精品人人爽 | 午夜视频久久久久一区| 日韩欧美亚洲综合久久影院d3| 国产69精品久久久久久人妻精品| 中文字幕久久精品| 内射无码专区久久亚洲| 久久久久精品国产亚洲AV无码| 99久久国产综合精品女同图片| 久久SE精品一区二区| 久久99国产综合精品| 久久久久久A亚洲欧洲AV冫| 亚洲AV无码久久| 久久久久久国产a免费观看黄色大片 | 欧美粉嫩小泬久久久久久久 | 久久精品一区二区三区AV|