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

那誰的技術博客

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

[算法問題]尋找一個序列中第n大的元素

問題描述:給定一個序列,以及指定這個序列的一個范圍,尋找這個范圍之內第n大的元素,如果n大于這個范圍之內的元素數量那么就返回-1.

這是快速排序算法中partiton算法的一個應用,不斷的分割序列,如果分割的位置正好是要找的位置,那么返回結果,否則視情況在前半部分和后半部分繼續查找,當然這個時候n值也要相應的變化了~~

/* *******************************************************************
????created:????2006/07/04
????filename:?????nthElement.cpp
????author:????????李創
????????????????
http://www.shnenglu.com/converse/

????purpose:????得到一個序列某個范圍以內的第n個元素的算法演示
????????????????提供了這個算法的遞歸和非遞歸的實現算法
????????????????同時為了測試之用提供了堆算法,用于在找到第N個元素之后
????????????????和排序之后的數組對應位置元素進行比較以測試代碼是否正確
********************************************************************
*/


#include?
< stdio.h >
#include?
< stdlib.h >
#include?
< time.h >

// ?交換元素
void ?Swap( int ? * a,? int ? * b)
{
????
int ?temp;

????temp?
= ? * a;
????
* a??? = ? * b;
????
* b??? = ?temp;
}


// ?打印數組元素
void ?DisplayArray( int ?array[],? int ?length)
{
????
int ?i;

????
for ?(i? = ? 0 ;?i? < ?length;?i ++ )
????
{
????????printf(
" array[%d]?=?%d\n " ,?i,?array[i]);
????}

}


// ?隨機創建一個數組
void ?CreateNewArray( int ?array[],? int ?length)
{
????
for ?( int ?i? = ? 0 ;?i? < ?length;?i ++ )
????
{
????????array[i]?
= ?rand()? % ? 256 ;
????}

}


// ?對一個給定范圍的子序列選定一個樞紐元素,執行完函數之后返回分割元素所在的位置,
// ?在分割元素之前的元素都小于樞紐元素,在它后面的元素都大于這個元素
int ?Partition( int ?array[],? int ?low,? int ?high)
{
????
// ?采用子序列的第一個元素為樞紐元素
????
// ?非常奇怪,如果我把選擇樞紐元素的算法改成注釋掉的那一行那么就會出錯(不是必現的)
????
// ?難道樞紐元素的選擇也會對這個算法產生影響?
????
// ?我在快速排序算法中測試了這個函數,兩種選擇樞紐元素的算法最后得到的結果都是正確的~~
????
// int?pivot?=?array[(low?+?high)?/?2];
???? int ?pivot? = ?array[low];

????
while ?(low? < ?high)
????
{
????????
// ?從后往前在后半部分中尋找第一個小于樞紐元素的元素
???????? while ?(low? < ?high? && ?array[high]? >= ?pivot)
????????
{
????????????
-- high;
????????}


????????
// ?將這個比樞紐元素小的元素交換到前半部分
????????Swap( & array[low],? & array[high]);

????????
// ?從前往后在前半部分中尋找第一個大于樞紐元素的元素
???????? while ?(low? < ?high? && ?array[low]? <= ?pivot)
????????
{
????????????
++ low;
????????}


????????
// ?將這個比樞紐元素大的元素交換到后半部分
????????Swap( & array[low],? & array[high]);
????}


????
// ?返回樞紐元素所在的位置
???? return ?low;
}


// ?尋找數組array中區間為[low,?high]中的第n大的元素,
// ?如果n大于這個區間的元素個數就返回-1
// ?非遞歸實現,這個非遞歸的實現很是簡單,就是把幾個出口的遞歸調用改寫成循環就好了~~
int ?nthElement2( int ?array[],? int ?low,? int ?high,? int ?n)
{
????
if ?(low? > ?high? || ?n? < ? 1 ? || ?n? > ?high? - ?low? + ? 1 )
????
{
????????
return ? - 1 ;
????}


????
int ?i;
????
while ?( 1 )
????
{
????????i?
= ?Partition(array,?low,?high);

????????
if ?(low? + ?n? - ? 1 ? == ?i)
????????
{
????????????
return ?array[i];
????????}

????????
else ? if ?(low? + ?n? - ? 1 ? < ?i)
????????
{
????????????
// return?nthElement(array,?low,?i?-?1,?n);
????????????high? = ?i? - ? 1 ;
????????}

????????
else ? if ?(low? + ?n? - ? 1 ? > ?i)
????????
{
????????????
// return?nthElement(array,?i?+?1,?high,?n?-?(i?-?low?+?1));
????????????low? = ?i? + ? 1 ;
????????????n?
= ?n? - ?(i? - ?low? + ? 2 );
????????}

????}

}


// ?尋找數組array中區間為[low,?high]中的第n大的元素,
// ?如果n大于這個區間的元素個數就返回-1
// ?遞歸實現
int ?nthElement( int ?array[],? int ?low,? int ?high,? int ?n)
{
????
if ?(low? > ?high? || ?n? < ? 1 ? || ?n? > ?high? - ?low? + ? 1 )
????
{
????????
return ? - 1 ;
????}


????
int ?i? = ?Partition(array,?low,?high);

????
if ?(low? + ?n? - ? 1 ? == ?i)
????
{
????????
return ?array[i];
????}

????
else ? if ?(low? + ?n? - ? 1 ? < ?i)
????
{
????????
return ?nthElement(array,?low,?i? - ? 1 ,?n);
????}

????
else ? if ?(low? + ?n? - ? 1 ? > ?i)
????
{
????????
return ?nthElement(array,?i? + ? 1 ,?high,?n? - ?(i? - ?low? + ? 1 ));
????}

}


// ?調整堆數組
// ?array是待調整的堆數組,i是待調整的數組元素的位置,length是數組的長度
void ?HeapAdjust( int ?array[],? int ?i,? int ?length)
{
????
int ?child,?temp;

????
for ?(temp? = ?array[i];? 2 ? * ?i? + ? 1 ? < ?length;?i? = ?child)
????
{
????????child?
= ? 2 ? * ?i? + ? 1 ;

????????
// ?得到子結點中較小的結點
???????? if ?(child? != ?length? - ? 1 ? && ?array[child? + ? 1 ]? > ?array[child])
????????????
++ child;

????????
// ?如果較小的子結點大于父結點那么把較小的子結點往上移動,替換它的父結點
???????? if ?(temp? < ?array[child])
????????
{
????????????array[i]?
= ?array[child];
????????}

????????
else ???? // ?否則退出循環
???????? {
????????????
break ;
????????}

????}


????
// ?最后把需要調整的元素值放到合適的位置
????array[i]? = ?temp;
}

// ?堆排序算法
void ?HeapSort( int ?array[],? int ?length)
{
????
// ?調整序列的前半部分元素,調整完之后第一個元素是序列的最大的元素
???? for ?( int ?i? = ?length? / ? 2 ? - ? 1 ;?i? >= ? 0 ;? -- i)
????
{
????????HeapAdjust(array,?i,?length);
????}


????
// ?從最后一個元素開始對序列進行調整,不斷的縮小調整的范圍直到第一個元素
???? for ?( int ?i? = ?length? - ? 1 ;?i? > ? 0 ;? -- i)
????
{
????????
// ?把第一個元素和當前的最后一個元素交換,
????????
// ?保證當前的最后一個位置的元素都是在現在的這個序列之中最大的
????????Swap( & array[ 0 ],? & array[i]);

????????
// ?對當前的序列進行調整,調整完之后保證第一個元素是當前序列的最大值
????????HeapAdjust(array,? 0 ,?i);
????}

}


int ?main( void )
{
????
int ?array[ 10 ];
????srand(time(NULL));

????
int ?n;
????printf(
" input?n:\n " );
????scanf(
" %d " ,? & n);

????
// ?測試遞歸程序
????printf( " 測試算法的遞歸函數實現\n " );
????CreateNewArray(array,?
10 );
????DisplayArray(array,?
10 );
????
int ?i? = ?nthElement(array,? 0 ,? 9 ,?n);
????
????HeapSort(array,?
10 );
????printf(
" after?Heap?Sort:\n " );
????DisplayArray(array,?
10 );

????printf(
" \nfind?%d?=?%d\n " ,?n,?i);
????
if ?(array[n? - ? 1 ]? == ?i)
????
{
????????printf(
" found!!\n " );
????}


????
// ?測試非遞歸函數的實現
????printf( " 測試算法的遞歸函數實現\n " );
????CreateNewArray(array,?
10 );
????DisplayArray(array,?
10 );
????i?
= ?nthElement2(array,? 0 ,? 9 ,?n);

????HeapSort(array,?
10 );
????printf(
" after?Heap?Sort:\n " );
????DisplayArray(array,?
10 );

????printf(
" \nfind?%d?=?%d\n " ,?n,?i);
????
if ?(array[n? - ? 1 ]? == ?i)
????
{
????????printf(
" found!!\n " );
????}



????system(
" pause " );

????
return ? 0 ;
}

posted on 2006-07-08 02:04 那誰 閱讀(3342) 評論(8)  編輯 收藏 引用 所屬分類: 算法與數據結構

評論

# re: [算法問題]尋找一個序列中第n大的元素  回復  更多評論   

使用STL里面的partial_sort很簡單的就可以得到一個序列中第N大的元素了。
2006-07-08 10:29 | 3×7=51

# re: [算法問題]尋找一個序列中第n大的元素  回復  更多評論   

犯錯誤了,使用STL里面的nth_element才是最好的辦法
2006-07-08 10:59 | 3×7=51

# re: [算法問題]尋找一個序列中第n大的元素  回復  更多評論   

我不否認有現成的可以用,但是我更愿意去了解其中的原理~~
2006-07-08 12:38 | 創系

# re: [算法問題]尋找一個序列中第n大的元素  回復  更多評論   

恕我直言,我覺得你這種方法并不是個好方法。等有時間我跟你討論一下這個問題還有什么解決方案。不過你可以去看下stl的實現(我自己還沒看)。
2006-07-08 16:55 | 3×7=51

# re: [算法問題]尋找一個序列中第n大的元素  回復  更多評論   

贊樓主自己思考問題:)
2007-03-30 14:40 | wtommy

# re: [算法問題]尋找一個序列中第n大的元素  回復  更多評論   

// int pivot = array[(low + high) / 2];
采用這句的時候會出問題,在調試中發現,當pivot本身就是數組中最大值時,low和high的會全部循環完,而跳出循環,沒有實現交換的目的;
可能是對算法理解有問題,不應該是low和high進行交換,應該是是和pivot進行交換
2007-06-18 20:14 | stream

# re: [算法問題]尋找一個序列中第n大的元素  回復  更多評論   

我覺得stream說的對,應該是和pivot交換。
2007-12-04 05:08 | alexandercer

# re: [算法問題]尋找一個序列中第n大的元素[未登錄]  回復  更多評論   

用紅黑樹數組,在紅黑樹中的每個節點都加入左邊和有邊的節點的數目,如果比左節點數目大就向左走,比右邊節點數目大,就減去這個數字再向右走,這樣一直走下去就找到了
2015-02-01 22:25 | shawn
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            久久嫩草精品久久久精品一| 永久555www成人免费| 久久综合色天天久久综合图片| 免费欧美高清视频| 久久成人精品| 欧美日韩一区二区视频在线观看 | 亚洲成色777777女色窝| 一区二区三区 在线观看视| 亚洲福利一区| 久久国产成人| 久久精品国产亚洲aⅴ| 欧美色综合网| 亚洲美女视频| 亚洲三级电影全部在线观看高清| 欧美一区二区三区日韩视频| 亚洲欧美视频在线观看视频| 欧美大片在线观看| 欧美激情导航| 亚洲精品国产精品国自产观看浪潮| 欧美在线观看网址综合| 欧美一区二区精品在线| 欧美视频一区二区三区在线观看 | 亚洲欧美电影院| 亚洲网站视频| 欧美日本中文| 99视频超级精品| 亚洲——在线| 国产精品女人久久久久久| 亚洲精选视频免费看| 亚洲精品影院| 欧美精品亚洲一区二区在线播放| 亚洲国产一区二区精品专区| 狠狠爱www人成狠狠爱综合网| 久久国产精品毛片| 久久综合九色| 亚洲电影欧美电影有声小说| 久久久精品五月天| 牛牛国产精品| 日韩视频在线一区| 欧美揉bbbbb揉bbbbb| 亚洲图片自拍偷拍| 欧美在线视频观看免费网站| 国产女精品视频网站免费| 欧美一区二区三区四区在线| 老色批av在线精品| 亚洲国产精品成人精品| 欧美精品一区二区三区在线播放 | 午夜精品婷婷| 麻豆国产va免费精品高清在线| 在线观看亚洲专区| 欧美激情精品久久久久久蜜臀 | 国产精品久久久久免费a∨| 亚洲男人的天堂在线aⅴ视频| 久久成人精品电影| 亚洲激情网站| 欧美日韩综合视频网址| 午夜久久资源| 欧美成人午夜激情| 亚洲午夜电影网| 国产一区二区三区四区hd| 美女在线一区二区| 亚洲网站在线观看| 欧美激情一区二区三区 | 亚洲精品中文字幕在线| 国产精品国码视频| 久久久久久久激情视频| 亚洲日韩欧美视频| 久久精品色图| 一区二区高清视频| 狠狠综合久久av一区二区小说| 欧美成人a视频| 亚洲欧美综合v| 亚洲国产欧美一区| 久久久99精品免费观看不卡| 亚洲欧洲精品一区二区精品久久久 | 精品91视频| 欧美日韩一区二| 久久经典综合| 夜夜精品视频一区二区| 欧美va亚洲va香蕉在线| 亚洲自拍高清| 亚洲美女av在线播放| 国产自产女人91一区在线观看| 欧美日韩国语| 开心色5月久久精品| 午夜一区二区三区在线观看| 亚洲日本视频| 欧美高清在线一区| 久久久久成人精品| 午夜精品久久久久久久男人的天堂| 亚洲高清在线视频| 国产一区二区成人久久免费影院| 欧美日韩在线精品一区二区三区| 久久青草欧美一区二区三区| 亚洲免费视频中文字幕| 亚洲精品乱码久久久久久日本蜜臀| 久久视频免费观看| 欧美一区二区免费观在线| 中日韩高清电影网| 亚洲精品日产精品乱码不卡| 精品福利电影| 激情久久五月| 国产一区二区三区精品欧美日韩一区二区三区 | 另类亚洲自拍| 久久蜜桃香蕉精品一区二区三区| 欧美一级在线亚洲天堂| 亚洲综合99| 亚洲欧美视频一区| 午夜精品网站| 欧美在线免费视频| 久久精品久久综合| 久久久国产视频91| 久久人人97超碰精品888| 欧美中文字幕在线观看| 久久国产色av| 久久看片网站| 美女爽到呻吟久久久久| 免费不卡亚洲欧美| 欧美激情91| 欧美丝袜第一区| 国产精品激情av在线播放| 国产精品伦理| 国产日韩欧美综合| 在线观看福利一区| 亚洲欧洲日韩在线| 日韩一级视频免费观看在线| 99视频有精品| 亚洲欧美成人网| 久久99伊人| 久久综合狠狠| 亚洲国产欧美另类丝袜| 一本色道久久| 亚洲欧美日韩国产中文| 久久国产精品一区二区| 久久最新视频| 欧美午夜激情小视频| 国产三级欧美三级日产三级99| 激情综合网址| 在线视频你懂得一区二区三区| 亚洲一区二区三区在线播放| 欧美一区午夜精品| 欧美顶级艳妇交换群宴| 亚洲精品国产视频| 亚洲欧美999| 久久全球大尺度高清视频| 欧美人妖在线观看| 国产在线不卡视频| 亚洲精品综合精品自拍| 午夜精品福利在线| 欧美成人影音| 亚洲一区二区三区欧美 | 久久av最新网址| 欧美激情亚洲综合一区| 国产欧美精品xxxx另类| 最新高清无码专区| 午夜一区不卡| 亚洲第一精品夜夜躁人人爽| 日韩视频欧美视频| 久久人人爽人人| 国产精品久久久久婷婷| 亚洲国产天堂网精品网站| 亚洲欧美一区二区在线观看| 欧美丰满高潮xxxx喷水动漫| 一区二区三区四区在线| 久久人人爽人人爽| 国产精品主播| 99精品欧美一区| 久久一区二区三区四区五区| 日韩一二三在线视频播| 噜噜噜躁狠狠躁狠狠精品视频 | 亚洲在线视频观看| 欧美国产日韩精品| 亚洲欧美国产三级| 欧美偷拍一区二区| 亚洲日韩成人| 欧美福利精品| 久久国产主播| 国产日韩欧美另类| 亚洲欧美日韩综合一区| 亚洲精品日日夜夜| 欧美黄色免费| 91久久综合| 欧美多人爱爱视频网站| 欧美综合激情网| 国产日韩专区在线| 欧美亚洲视频在线观看| 一本色道久久| 国产精品高潮呻吟| 中文一区二区| 日韩午夜视频在线观看| 欧美片第一页| 一本色道久久综合亚洲精品小说| 欧美成人激情视频| 久久天堂av综合合色| 韩国v欧美v日本v亚洲v| 久久久免费精品视频| 欧美一级一区| 好吊一区二区三区| 麻豆精品在线观看| 狂野欧美一区|