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

那誰的技術博客

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

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

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

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

/* *******************************************************************
????created:????2006/07/04
????filename:?????nthElement.cpp
????author:????????李創
????????????????
http://m.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 那誰 閱讀(3339) 評論(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>
            国产一区久久| 久久嫩草精品久久久精品一| 欧美成人亚洲| 亚洲一区二区久久| 亚洲国产一二三| 欧美在线视频a| 欧美一区二区三区四区视频| 亚洲日产国产精品| 亚洲国产毛片完整版| 亚洲二区三区四区| 亚洲国产电影| 亚洲天堂网在线观看| 欧美中文在线观看| 欧美成人国产| 亚洲第一页在线| 亚洲高清视频在线| 中日韩午夜理伦电影免费| 亚洲一区免费网站| 久久久精彩视频| 欧美成人黑人xx视频免费观看| 开心色5月久久精品| 欧美激情五月| 国产在线视频欧美一区二区三区| 在线观看亚洲专区| 亚洲乱码国产乱码精品精98午夜| 亚洲国产高清视频| 香蕉成人啪国产精品视频综合网| 欧美日韩一区在线观看视频| 欧美三级精品| 亚洲精品免费观看| 日韩一区二区高清| 亚洲图片在线观看| 欧美亚洲综合在线| 国产精品视频xxx| 亚洲欧美经典视频| 亚洲免费高清| 欧美精品激情在线观看| 亚洲欧洲精品一区二区三区不卡 | 欧美一区国产二区| 最新成人av在线| 日韩一二三区视频| 免费毛片一区二区三区久久久| 欧美午夜欧美| 亚洲欧美激情四射在线日 | 久久亚洲综合网| 国产视频久久久久| 免费的成人av| 欧美国产精品劲爆| 这里只有视频精品| 欧美.www| 欧美日韩在线第一页| 亚洲欧洲一区二区天堂久久| 久久在线视频| 亚洲第一网站免费视频| 奶水喷射视频一区| 欧美xart系列在线观看| aaa亚洲精品一二三区| 亚洲天堂视频在线观看| 精品1区2区| 亚洲第一页在线| 国产区二精品视| 久久aⅴ国产紧身牛仔裤| 久久综合色一综合色88| 亚洲乱亚洲高清| 国产精品亚洲欧美| 欧美大片网址| 在线日韩欧美视频| 欧美成人精品福利| 国产日韩欧美三级| 一区二区欧美日韩视频| 亚洲国产毛片完整版| 久久精品在线免费观看| 午夜国产一区| 国产精品久久久久久一区二区三区| 欧美在线亚洲在线| 欧美日韩精品免费观看视频完整 | 国产伦精品一区二区| 亚洲国产一区二区视频| 国产日本欧美在线观看| 亚洲私人黄色宅男| 一区二区三区日韩在线观看| 免费h精品视频在线播放| 久久国产精品99国产| 亚洲精品社区| 欧美成人一区二区在线| 欧美中文字幕视频在线观看| 欧美日韩国产精品成人| 亚洲精品久久久蜜桃 | 久久精品人人| 精品成人一区二区| 欧美成人黑人xx视频免费观看| 欧美成人嫩草网站| 亚洲麻豆一区| 国产一区二区三区久久悠悠色av| 亚洲电影中文字幕| 欧美激情在线免费观看| 亚洲精品国产欧美| 午夜在线视频一区二区区别| 国产欧美一区二区精品婷婷| 久久婷婷国产综合精品青草| 久久综合九色九九| 亚洲人成在线观看网站高清| 欧美激情精品| 欧美一区视频在线| 亚洲精品女av网站| 制服丝袜激情欧洲亚洲| 国产一区二区日韩精品| 麻豆国产精品777777在线| 国产精品久久久久久久久借妻| 一本久久知道综合久久| 久久蜜桃精品| 久久不射电影网| 亚洲精品少妇30p| 亚洲精品永久免费| 国产亚洲观看| 国产精品久久久久一区二区三区共| 麻豆久久精品| 欧美大片国产精品| 免费亚洲一区二区| 欧美成人嫩草网站| 欧美日韩精品欧美日韩精品一| 欧美在线观看天堂一区二区三区| 亚洲成人中文| 国产美女精品人人做人人爽| 亚洲你懂的在线视频| 99精品视频网| 免费观看国产成人| 欧美亚洲系列| 久久国产精品亚洲va麻豆| 午夜精品一区二区三区在线| 亚洲六月丁香色婷婷综合久久| 最新国产成人在线观看| 亚洲青色在线| 午夜精品久久久久久久久| 亚洲综合欧美| 久久久久久夜精品精品免费| 久久久久.com| 亚洲三级视频| 久久精品国产免费看久久精品| 亚洲午夜av在线| 欧美在线国产| 欧美日韩在线播放一区| 国产在线播放一区二区三区| 亚洲高清成人| 亚洲欧美日韩一区| 亚洲高清视频的网址| 亚洲自拍偷拍网址| 欧美高清日韩| 国内在线观看一区二区三区| 99热在这里有精品免费| 久久一区二区三区av| 一区二区三欧美| 免费欧美日韩| 激情欧美国产欧美| 亚洲欧美日本国产专区一区| 久久香蕉国产线看观看av| 99re亚洲国产精品| 久久一区精品| 亚洲黄色一区二区三区| 久久综合色影院| 国产精品99久久久久久久女警| 亚洲欧美日本日韩| 中文一区二区在线观看| 欧美日韩18| 亚洲性线免费观看视频成熟| 亚洲电影免费观看高清完整版| 久久久www成人免费无遮挡大片| 嫩草伊人久久精品少妇av杨幂| 国产一区二区在线免费观看| 久久精品在线观看| 欧美大成色www永久网站婷| 亚洲美女免费精品视频在线观看| 你懂的国产精品永久在线| 裸体一区二区三区| 亚洲一区二区毛片| 欧美一级日韩一级| 亚洲精品久久久久久一区二区| 亚洲日本va午夜在线电影| 国产精品一区毛片| 欧美电影在线观看完整版| 国产精品剧情在线亚洲| 蜜桃精品一区二区三区| 久久夜色精品| 欧美在线视频播放| 另类亚洲自拍| 欧美伊人久久大香线蕉综合69| 欧美亚洲系列| 亚洲综合精品| 欧美精品综合| 欧美国产综合视频| 国产欧美在线播放| 亚洲视频欧洲视频| 亚洲毛片在线观看.| 先锋a资源在线看亚洲| 亚洲字幕在线观看| 欧美性色aⅴ视频一区日韩精品| 美女主播一区| 黄色亚洲在线| 在线亚洲美日韩| 亚洲一区二区三区免费在线观看|