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

那誰的技術博客

感興趣領域:高性能服務器編程,存儲,算法,Linux內核
隨筆 - 210, 文章 - 0, 評論 - 1183, 引用 - 0
數據加載中……

原地歸并算法

歸并排序算法(mergesort)是將一個序列劃分為同樣大小的兩個子序列,然后對兩個子序列分別進行排序,最后進行合并操作,將兩個子序列合成有序的序列.在合成的過程中,一般的實現都需要開辟一塊與原序列大小相同的空間,以進行合并操作,歸并排序算法的示例在這里.

這里介紹一種不需要開辟新的空間就可以進行歸并操作的算法.算法的核心部分是以下代碼:
 1 /**
 2 * 算法: 合并二已排序的連續序列
 3 **/
 4 template<typename T>
 5 void t_merge( T& v, size_t size, size_t pos )
 6 {
 7     size_t fir = 0, sec = pos;
 8     while ( fir < sec && sec < size )
 9     {
10         while ( fir < sec && v[fir] <= v[sec] ) fir++;
11         size_t maxMove = 0;
12         while ( sec < size && v[fir] > v[sec] ) maxMove++, sec++;
13         t_exchange( &v[fir], sec - fir, sec - fir - maxMove );
14         fir += maxMove;       
15     }
16 }

其中T是一個數組, size是數組尺寸, pos是歸并劃分的位置.也就是說[0,pos)和[pos, size)都分別是有序的.比如對序列1, 3, 5, 7, 2, 4, 6, 8進行歸并操作, 此時size=8, pos = 4.

以<<算法導論>>中介紹的通過循環不變量的方法證明算法的正確性,在這個算法中, 循環不變量為[fir, sec)中的元素都是有序的:

1) 初始:此時fir = 0, sec = pos, 以前面介紹的函數參數的說明來看,滿足循環不變量.

2) 迭代:來看看循環做了些什么操作.行10進行的操作為, 只要滿足fir元素不大于sec元素, fir就一直遞增;行12進行的操作是只要滿足fir大于sec, sec就一直遞增, 同時遞增maxMove計數.因此, 進行完前面兩個步驟之后, fir所指元素一定小于sec以及其后的所有元素.而位于sec之前的第二個子序列中的元素, 一定小于fir.因此, [sec-maxMove, sec)z中的元素小于所有[fir, sec - 1)的元素.通過調用t_exchange函數將位于[sec-maxMove, sec)中的元素"旋轉"到fir之前.

也就是說, 這個過程在第二個已經排序好的子序列中尋找在它之內的小于目前第一個已經排序好的子序列的序列, 將它"旋轉"到前面.

以序列 1, 3, 5, 7, 2, 4, 6, 8為例, 此時fir=1也就是指向3, sec=5也就是指向4, maxMove=1, 通過調用t_exchange函數之后將[sec-maxMove, sec)即[4,5)中的元素也就是2"旋轉"到子序列3,5,7之前,于是該循環結束之后序列變為1,2,3,5,7,4,6,8, 此時fir=2, sec =5, 滿足循環不變量.

3) 終止: 當循環終止時, fir或者sec之一超過了數組的尺寸, 顯而易見, 此時序列變成了有序的.

完整的算法如下所示, 需要特別說明的是, 這段代碼不是我想出來的, 原作者在這里:
#include <stdio.h>
#include 
<iostream>

using namespace std;

//int array[] = {1, 3, 5, 7, 2, 4, 6, 8};
int array[] = {3,5,7,8,1,2,4,6};
void display(int array[], int n)
{
     
for (int i = 0; i < n; ++i)
     {
         printf(
"%d ", array[i]);
     }
     printf(
"\n");
}

/**
* 算法: 交換二對象
*
*/
template
<typename T>
void t_swap( T& v1, T& v2 )
{
    T t 
= v1; v1 = v2; v2 = t;
}

/**
* 算法: 反轉序列
*
*/
template
<typename T>
void t_reverse( T* v, size_t size )
{
    size_t s 
= 0, e = size-1;
    
while( s < e && s < size && e > 0 )
        t_swap( v[s
++], v[e--] );
}

/**
* 算法: 手搖算法,從指定位置旋轉序列(見編程珠璣第二章)
*
*/
template
<typename T>
void t_exchange( T* v, size_t size, size_t n )
{
    t_reverse( v, n );
    t_reverse( v 
+ n, size - n );
    t_reverse( v, size );
}

/**
* 算法: 合并二已排序的連續序列
*
*/
template
<typename T>
void t_merge( T& v, size_t size, size_t pos )
{
    size_t fir 
= 0, sec = pos;
    
while ( fir < sec && sec < size )
    {
        
while ( fir < sec && v[fir] <= v[sec] ) fir++;
        size_t maxMove 
= 0;
        
while ( sec < size && v[fir] > v[sec] ) maxMove++, sec++;
        t_exchange( 
&v[fir], sec - fir, sec - fir - maxMove );
        fir 
+= maxMove;

        display(array, 
sizeof(array)/sizeof(int));
    }
}

/**
* 算法: 歸并排序
*
*/
template
<typename T>
void t_merge_sort( T* v, size_t size )
{
    
if ( size <= 1 ) return;
    t_merge_sort( v, size
/2 );
    t_merge_sort( v 
+ size/2, size - size/2 );
    t_merge( v, size, size
/2 );
}

int main()
{
     display(array, 
sizeof(array)/sizeof(int));

     t_merge(array, 
sizeof(array) / sizeof(int), (sizeof(array)/sizeof(int))/2);
     
//t_merge_sort(array, sizeof(array) / sizeof(int));

     display(array, 
sizeof(array)/sizeof(int));
     
return 0;
}


補充說明:
其實前面采用"旋轉"算法將元素前移不是必須的, 可以將所要移動的元素之前的部分后移, 再將元素插入到合適的位置.比如前面說的對于序列1, 3, 5, 7, 2, 4, 6, 8, 第一步要將元素2前移至3之前, 可以將3,5,7后移, 然后將2插入到合適的位置.
但是這樣有一個問題, 如果要移動的元素多了, 那么就需要更多的臨時空間保存要前移的元素, 這樣對空間就不是O(1)的了.而旋轉算法可以做到O(1)的空間達到要求.


posted on 2008-09-28 19:51 那誰 閱讀(5976) 評論(7)  編輯 收藏 引用 所屬分類: 算法與數據結構

評論

# re: 原地歸并算法  回復  更多評論   

呵呵 你對算法好有研究哇
2008-09-28 20:54 | 沈臻豪(foxtail)

# re: 原地歸并算法  回復  更多評論   

“通過調用t_exchange函數將位于[sec-maxMove+fir, sec)中的元素"旋轉"到fir之前.”——區間左端多了一個+fir

真是非常棒的對歸并的改進!在較好的情況下,合并操作的時間開銷會降低;而在最壞情況下,合并操作的開銷從操作數量上來說是傳統歸并的三倍(swap相當于執行三次賦值),但是考慮傳統歸并中內存分配的時間,實踐中估計是差不了太多。然而最關鍵在于這是個原地算法。
希望看到博主更多的算法研究文章。
2008-09-29 10:52 | E劍仙

# re: 原地歸并算法  回復  更多評論   

@E劍仙
感謝指出錯誤,已經修改!
2008-09-29 15:48 |

# re: 原地歸并算法  回復  更多評論   

重新看了下這個算法,發現補充說明那也不對,用插入+整體偏移的方法實現也是能只需要O(1)的空間,不會比旋轉算法差
2008-11-13 22:05 | E劍仙

# re: 原地歸并算法  回復  更多評論   

插入+整體偏移 對于數組來說是O(N2)的,旋轉是O(N)的。。
當然空間是一樣的
2010-07-08 19:00 | laowan

# re: 原地歸并算法  回復  更多評論   

swap相當于執行三次賦值 其總的賦值平均時間代價為6n方 其代價反而高于最簡單的挪動的原地歸并(最簡單的挪動的賦值平均時間代價為0.5n方)
2012-05-07 21:42 | Glawind

# re: 原地歸并算法  回復  更多評論   

原地歸并算法存在時間復雜度為O(n)空間復雜度為O(1)的實現
我就做了個
2013-08-08 08:24 | scof
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            亚洲欧美日韩系列| 亚洲一卡二卡三卡四卡五卡| 一本色道久久综合亚洲精品不卡| 一本色道久久加勒比88综合| 欧美午夜精品电影| 欧美一区二区三区免费观看视频| 亚洲人成网站在线播| 国模 一区 二区 三区| 久久精品国产久精国产一老狼| 亚洲欧美制服中文字幕| 中文一区二区在线观看| 午夜在线a亚洲v天堂网2018| 国产亚洲精品久久久久动| 欧美日韩免费网站| 欧美日韩伦理在线| 欧美视频福利| 国产主播一区二区三区| 狠狠色噜噜狠狠色综合久| 亚洲国产精品一区制服丝袜| 亚洲国产高清自拍| 亚洲精品三级| 久久成人精品无人区| 你懂的视频一区二区| 99亚洲视频| 久久久久久一区| 欧美四级电影网站| 黄色成人91| 久久成年人视频| 日韩亚洲在线观看| 中日韩美女免费视频网址在线观看 | 可以看av的网站久久看| 久久久久成人精品| 亚洲激情小视频| 性久久久久久久久久久久| 欧美成人精品1314www| 国产精品系列在线| 一区二区成人精品| 亚洲精品日韩激情在线电影| 久久精品一区二区三区不卡| 国产精品久久久久久久久免费桃花 | 久久婷婷激情| 亚洲一区二区三区四区视频| 欧美精品一二三| 亚洲人成77777在线观看网| 久久综合九色欧美综合狠狠| 午夜精品www| 加勒比av一区二区| 欧美福利一区| 久久综合狠狠综合久久激情| 亚洲一区二区三区高清| 亚洲电影一级黄| 免费不卡在线视频| 91久久在线视频| 亚洲久久成人| 国产农村妇女精品| 久久婷婷国产综合精品青草| 久久中文在线| 一区二区三区视频观看| 新67194成人永久网站| 在线免费观看一区二区三区| 亚洲成色999久久网站| 欧美日产在线观看| 欧美在线免费| 国产精品乱人伦一区二区| 一区二区欧美在线观看| 午夜日韩电影| 亚洲先锋成人| 欧美日本亚洲| 欧美高清在线观看| 国产综合色一区二区三区| 免费不卡欧美自拍视频| 国产精品久久二区| 久久先锋影音av| 国产精品成人一区二区| 亚洲激情视频在线播放| 欲色影视综合吧| 久久久久久九九九九| 性欧美在线看片a免费观看| 欧美人成网站| 日韩天堂在线观看| 99热免费精品| 欧美欧美天天天天操| 欧美成人小视频| 亚洲日本在线视频观看| 蜜臀91精品一区二区三区| 欧美黄色aa电影| 亚洲国产精品第一区二区| 免费视频一区| av成人免费在线| 亚洲欧美日韩国产一区| 国产毛片一区二区| 久久久久久亚洲精品杨幂换脸 | 亚洲在线观看| 久久久综合网站| 在线观看视频一区二区| 免费在线欧美视频| 亚洲一区免费网站| 亚洲美女黄色片| 国产精品分类| 久久只有精品| 亚洲视频国产视频| 国产精品亚洲一区| 久久久久久色| 午夜一区在线| 一本色道久久加勒比精品| 久久久久网址| 亚洲欧洲在线一区| 国产一区二区三区不卡在线观看| 另类天堂av| 久久精品成人欧美大片古装| 亚洲最新视频在线| 亚洲欧洲三级电影| 美女视频黄免费的久久| 亚洲女人av| 亚洲视频在线观看网站| 亚洲二区在线观看| 樱花yy私人影院亚洲| 国产综合精品| 国产一区二区毛片| 极品少妇一区二区三区| 国产视频观看一区| 国产老女人精品毛片久久| 国产精品久久综合| 国产精品久久亚洲7777| 国产精品美女www爽爽爽视频| 欧美日韩国内| 欧美午夜电影网| 国产欧美日韩精品专区| 国产日韩一区二区三区在线播放| 国产欧美日韩一区二区三区在线| 国产色爱av资源综合区| 国产一区二区无遮挡| 好看的日韩av电影| 9l国产精品久久久久麻豆| 亚洲欧美一区二区视频| 久久久爽爽爽美女图片| 91久久精品日日躁夜夜躁欧美| 欧美视频在线视频| 久久精品卡一| 欧美日韩视频免费播放| 国产欧美日韩精品丝袜高跟鞋| 亚洲国产高清在线| 欧美亚洲三区| 一区二区不卡在线视频 午夜欧美不卡'| 午夜精品国产精品大乳美女| 麻豆av一区二区三区| 国产精品素人视频| 亚洲三级电影在线观看| 久久久精品国产99久久精品芒果| 亚洲日本无吗高清不卡| 欧美亚洲一区二区在线观看| 欧美色综合网| 亚洲精品日本| 亚洲人成高清| 另类av导航| 亚洲成色精品| 欧美11—12娇小xxxx| 欧美在线国产精品| 国产麻豆9l精品三级站| 国产日韩欧美日韩大片| 欧美激情欧美激情在线五月| 好吊一区二区三区| 欧美一区二区三区啪啪| 午夜激情一区| 国产亚洲精品久久久| 久久深夜福利免费观看| 久久久成人精品| 亚洲国产精品久久精品怡红院| 欧美成人国产| 欧美理论电影在线观看| 亚洲香蕉在线观看| 亚洲欧美日韩在线不卡| 在线播放一区| 一本久久综合亚洲鲁鲁五月天| 国产精品尤物福利片在线观看| 久久一区二区三区四区五区| 欧美成人一区二区三区片免费| 一区二区三区欧美在线观看| 亚洲自拍偷拍色片视频| 亚洲国产精品一区二区尤物区 | 亚洲男人的天堂在线观看| 国产欧美成人| 99re8这里有精品热视频免费| 国产亚洲成年网址在线观看| 亚洲精品一区二区在线观看| 国产精品嫩草99a| 亚洲高清不卡在线| 国产欧美一区二区三区在线看蜜臀| 麻豆91精品| 国产亚洲免费的视频看| 亚洲午夜精品国产| 亚洲在线视频观看| 欧美精品v日韩精品v国产精品| 久久久一本精品99久久精品66| 欧美性jizz18性欧美| 亚洲经典三级| 一区二区三区产品免费精品久久75 | 国产精品高清网站| 一区二区激情视频| 亚洲国产欧美一区|