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

那誰的技術(shù)博客

感興趣領(lǐng)域:高性能服務(wù)器編程,存儲(chǔ),算法,Linux內(nèi)核
隨筆 - 210, 文章 - 0, 評(píng)論 - 1183, 引用 - 0
數(shù)據(jù)加載中……

原地歸并算法

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

這里介紹一種不需要開辟新的空間就可以進(jìn)行歸并操作的算法.算法的核心部分是以下代碼:
 1 /**
 2 * 算法: 合并二已排序的連續(xù)序列
 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是一個(gè)數(shù)組, size是數(shù)組尺寸, pos是歸并劃分的位置.也就是說[0,pos)和[pos, size)都分別是有序的.比如對(duì)序列1, 3, 5, 7, 2, 4, 6, 8進(jìn)行歸并操作, 此時(shí)size=8, pos = 4.

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

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

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

也就是說, 這個(gè)過程在第二個(gè)已經(jīng)排序好的子序列中尋找在它之內(nèi)的小于目前第一個(gè)已經(jīng)排序好的子序列的序列, 將它"旋轉(zhuǎn)"到前面.

以序列 1, 3, 5, 7, 2, 4, 6, 8為例, 此時(shí)fir=1也就是指向3, sec=5也就是指向4, maxMove=1, 通過調(diào)用t_exchange函數(shù)之后將[sec-maxMove, sec)即[4,5)中的元素也就是2"旋轉(zhuǎn)"到子序列3,5,7之前,于是該循環(huán)結(jié)束之后序列變?yōu)?,2,3,5,7,4,6,8, 此時(shí)fir=2, sec =5, 滿足循環(huán)不變量.

3) 終止: 當(dāng)循環(huán)終止時(shí), fir或者sec之一超過了數(shù)組的尺寸, 顯而易見, 此時(shí)序列變成了有序的.

完整的算法如下所示, 需要特別說明的是, 這段代碼不是我想出來的, 原作者在這里:
#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");
}

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

/**
* 算法: 反轉(zhuǎn)序列
*
*/
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--] );
}

/**
* 算法: 手搖算法,從指定位置旋轉(zhuǎn)序列(見編程珠璣第二章)
*
*/
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 );
}

/**
* 算法: 合并二已排序的連續(xù)序列
*
*/
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;
}


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


posted on 2008-09-28 19:51 那誰 閱讀(5981) 評(píng)論(7)  編輯 收藏 引用 所屬分類: 算法與數(shù)據(jù)結(jié)構(gòu)

評(píng)論

# re: 原地歸并算法  回復(fù)  更多評(píng)論   

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

# re: 原地歸并算法  回復(fù)  更多評(píng)論   

“通過調(diào)用t_exchange函數(shù)將位于[sec-maxMove+fir, sec)中的元素"旋轉(zhuǎn)"到fir之前.”——區(qū)間左端多了一個(gè)+fir

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

# re: 原地歸并算法  回復(fù)  更多評(píng)論   

@E劍仙
感謝指出錯(cuò)誤,已經(jīng)修改!
2008-09-29 15:48 | 創(chuàng)

# re: 原地歸并算法  回復(fù)  更多評(píng)論   

重新看了下這個(gè)算法,發(fā)現(xiàn)補(bǔ)充說明那也不對(duì),用插入+整體偏移的方法實(shí)現(xiàn)也是能只需要O(1)的空間,不會(huì)比旋轉(zhuǎn)算法差
2008-11-13 22:05 | E劍仙

# re: 原地歸并算法  回復(fù)  更多評(píng)論   

插入+整體偏移 對(duì)于數(shù)組來說是O(N2)的,旋轉(zhuǎn)是O(N)的。。
當(dāng)然空間是一樣的
2010-07-08 19:00 | laowan

# re: 原地歸并算法  回復(fù)  更多評(píng)論   

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

# re: 原地歸并算法  回復(fù)  更多評(píng)論   

原地歸并算法存在時(shí)間復(fù)雜度為O(n)空間復(fù)雜度為O(1)的實(shí)現(xiàn)
我就做了個(gè)
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>
            宅男精品视频| 国产无遮挡一区二区三区毛片日本| 欧美国产视频在线观看| 久久久水蜜桃| 免费成人美女女| 欧美韩日亚洲| 一本色道久久综合亚洲91| 亚洲午夜精品久久久久久浪潮 | 亚洲高清视频的网址| 久久深夜福利免费观看| 欧美jizzhd精品欧美巨大免费| 亚洲国产成人av| 亚洲视频一区| 久久麻豆一区二区| 欧美区视频在线观看| 国产欧美在线看| 亚洲精品美女在线观看| 欧美一区二区三区在线视频| 欧美岛国激情| 欧美刺激性大交免费视频| 亚洲高清在线播放| 亚洲在线观看视频网站| 久久精品中文字幕一区二区三区| 欧美不卡在线视频| 国产精品午夜av在线| 亚洲欧洲在线视频| 欧美一区亚洲二区| 亚洲麻豆av| 久久网站免费| 国产日韩精品一区二区三区在线| 亚洲激情女人| 久久久久国内| 亚洲图片欧美日产| 欧美激情一区在线| 在线免费观看日本一区| 亚洲一区二区三区免费在线观看 | 一本色道久久综合狠狠躁的推荐| 性欧美暴力猛交另类hd| 欧美日本在线视频| 尹人成人综合网| 久久成人免费网| 宅男噜噜噜66一区二区| 欧美国产日韩一区二区| **网站欧美大片在线观看| 久久激情视频久久| 亚洲伊人观看| 欧美午夜激情在线| 日韩亚洲在线观看| 欧美激情精品久久久久久免费印度 | 久久久久久亚洲精品中文字幕 | 一区二区三区黄色| 亚洲二区在线观看| 久久综合精品国产一区二区三区| 国产欧美韩日| 久久成人人人人精品欧| 亚洲欧美日韩区| 国产精品夜夜夜| 午夜久久tv| 亚洲欧美综合精品久久成人| 国产欧美日韩三区| 欧美在线短视频| 香蕉成人啪国产精品视频综合网| 国产精品你懂的在线| 午夜亚洲福利在线老司机| 在线视频欧美精品| 国产精品久久久久久久久久直播 | 欧美精品免费视频| 99re这里只有精品6| 亚洲国产高清自拍| 男人插女人欧美| 亚洲精品资源| 一本久道综合久久精品| 国产精品久久久久久久久久ktv| 亚洲在线播放电影| 午夜天堂精品久久久久| 激情综合在线| 亚洲国产高清在线观看视频| 欧美日韩视频免费播放| 小黄鸭精品密入口导航| 欧美在线国产| 亚洲人成人一区二区三区| 亚洲精一区二区三区| 国产欧美一区二区三区视频| 久久久夜色精品亚洲| 暖暖成人免费视频| 亚洲一区三区在线观看| 欧美中文字幕视频在线观看| 亚洲人成7777| 亚洲一区中文字幕在线观看| 激情欧美日韩一区| 99pao成人国产永久免费视频| 国产精品少妇自拍| 欧美成人精品在线| 国产精品护士白丝一区av| 久久综合综合久久综合| 欧美日韩在线一区二区| 另类成人小视频在线| 欧美色区777第一页| 另类国产ts人妖高潮视频| 国产精品vvv| 欧美激情一区二区三区在线| 国产精品久久777777毛茸茸| 欧美不卡三区| 国产视频在线观看一区二区| 亚洲日本欧美| 一区在线影院| 亚洲一级特黄| 一区二区久久久久| 久久综合久久综合久久综合| 亚洲欧美美女| 欧美激情综合亚洲一二区| 久久人体大胆视频| 国产精品日本精品| 亚洲伦理自拍| 日韩午夜激情av| 免费观看成人www动漫视频| 欧美在线视频观看| 欧美午夜精品久久久| 亚洲国产精品123| 在线播放日韩欧美| 性欧美大战久久久久久久免费观看 | 亚洲精品久久在线| 久久精品视频99| 欧美一区二区三区精品| 欧美午夜精品久久久久久浪潮| 亚洲激情不卡| 亚洲国产成人91精品| 欧美午夜a级限制福利片| 最近看过的日韩成人| 亚洲福利一区| 久久久久久免费| 久久久久久久久久久一区 | 一区二区三区视频免费在线观看| 亚洲欧洲一区二区在线观看| 久热re这里精品视频在线6| 久久亚洲精品一区二区| 国产偷自视频区视频一区二区| 亚洲一区二区三区乱码aⅴ蜜桃女| 亚洲欧美卡通另类91av| 国产精品嫩草99av在线| 亚洲欧美国产日韩中文字幕| 午夜伦欧美伦电影理论片| 国产精品久久久久久久久久三级 | 欧美国产一区在线| 亚洲国产三级网| 亚洲精品一区二区三区四区高清| 美女国产一区| 欧美风情在线观看| 亚洲精品在线三区| 欧美三区不卡| 亚洲天堂激情| 久久久久久久久久久一区| 在线观看视频免费一区二区三区| 久久久免费精品视频| 欧美黄色网络| 亚洲无限av看| 国产偷国产偷精品高清尤物| 久久偷窥视频| 夜夜嗨av一区二区三区免费区| 小嫩嫩精品导航| 亚洲国产一区二区三区高清 | 亚洲第一色在线| 欧美精品1区| 亚洲男同1069视频| 欧美激情乱人伦| 亚洲欧美精品| 在线国产亚洲欧美| 欧美午夜激情视频| 久久尤物视频| 亚洲视频在线观看| 欧美99久久| 亚洲欧美影院| 亚洲日本成人女熟在线观看| 国产精品极品美女粉嫩高清在线| 久久国产精品99精品国产| 亚洲人成小说网站色在线| 欧美在线播放视频| 一区二区三区日韩| 在线成人激情视频| 国产精品第2页| 另类专区欧美制服同性| 亚洲午夜羞羞片| 亚洲国产婷婷综合在线精品 | 黄网站免费久久| 欧美午夜精品理论片a级大开眼界| 久久国产精品99精品国产| 夜夜嗨av一区二区三区| 欧美顶级大胆免费视频| 久久久精品一品道一区| 亚洲激情婷婷| 久久精品视频免费播放| 中日韩美女免费视频网址在线观看 | 一区二区久久| 亚洲国产精品女人久久久| 久久av一区二区三区| 亚洲视频欧洲视频| 日韩午夜精品| 99亚洲伊人久久精品影院红桃| 国内免费精品永久在线视频| 欧美视频成人|