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

我所理解的堆排序算法
      堆排序在最壞的情況下,其時(shí)間復(fù)雜度也能達(dá)到O(nlogn)。相對(duì)于快速排序來說,這是它最大的優(yōu)點(diǎn),此外,堆排序僅需要一個(gè)記錄大小供交換用的輔助存儲(chǔ)空間。
      堆排序的數(shù)據(jù)結(jié)構(gòu)是二叉堆,二叉堆的特點(diǎn)有兩個(gè),一個(gè)是它是一棵完全二叉樹,另一個(gè)是它的根結(jié)點(diǎn)小于孩子結(jié)點(diǎn),所以我們很容易找到它的最小結(jié)點(diǎn)----根結(jié)點(diǎn);當(dāng)然如果你想找到最大結(jié)點(diǎn)的話,那就要掃描所有的葉子結(jié)點(diǎn),這是很費(fèi)時(shí)間的,如果你想找的是最大結(jié)點(diǎn)的話,你最好把它弄成一個(gè)大頂堆,即一棵根結(jié)點(diǎn)大于孩子結(jié)點(diǎn)的完全二叉樹。
      二叉堆通常用數(shù)組來實(shí)現(xiàn),它舍棄下標(biāo)0,從下標(biāo)1開始置數(shù),則很容易滿足,對(duì)于數(shù)組中任意位置i上的元素,其左兒子的位置在2i上,右兒子的位置在2i+1上,雙親的位置則在i/2上。
      堆排序的算法之一是把數(shù)組構(gòu)建成二叉堆----這只要增添一個(gè)長度為n+1的輔助空間,然后把原數(shù)組的元素依次插入到二叉堆即可。然后刪除二叉堆的根,把它作為排序后的數(shù)組的第一個(gè)元素,然后使二叉堆的長度減1,并通過上移使得新得到的序列仍為二叉堆,再提取新二叉堆的第一個(gè)元素到新數(shù)組。依此類推,直到提取最后一個(gè)元素,新得到的數(shù)組就是排序后的數(shù)組。
算法如下:
template <class T>
void Insert(T a[], int len, T x)//把x插入到原長度為len的二叉堆,注意保證新二叉堆不越界
{
      int i;
      for (i=len; i/2>0 && a[i/2]>x; i/=2)
            a[i] = a[i/2];
      a[i] = x;
}

template <class T>
T DeleteMin(T a[], int len)//刪除二叉堆的根,并通過上移使得新得到的序列仍為二叉堆
{
      if (len == 0)
            exit(1);
      T min = a[1];//二叉堆的根
      T last = a[len--];//二叉堆的最后一個(gè)元素

      int c;
      int i;
      for (i=1; i*2<=len; i=c)//把二叉堆的某些元素往前移,使得新得到的序列仍為二叉堆
      {
            c = i * 2;//i的左兒子
            if (c != len && a[c+1] < a[c])//若i有右兒子,且右兒子小于左兒子,c指向右兒子
                  c++;

            if (last > a[c])//若i的小兒子小于二叉堆的最后一個(gè)元素,把其移到i的位置
                  a[i] = a[c];
            else
                  break;
      }
      a[i] = last; //把二叉堆的最后一個(gè)元素放到適當(dāng)?shù)目瘴唬藭r(shí)得到的序列仍為二叉堆

      return min;
}

template <class T>
void HeapSort(T a[], int len)
{
      T *ca = new T[len+1]; //復(fù)制原數(shù)組到二叉堆
      ca[0] = 0;
      for (int i=0; i<len; i++) //把元素依次插入到二叉堆
            Insert(ca, i+1, a[i]);

      for (int i=0; i<len; i++)//依次提取二叉堆的根作為排序后的數(shù)組的元素
      {
            a[i] = DeleteMin(ca, len-i);
      }

      a[len-1] = ca[1]; //注意不能忘了最后一個(gè)元素

      delete []ca;
}
      在《數(shù)據(jù)結(jié)構(gòu)習(xí)題與解析》(李春葆 編著 清華大學(xué)出版社)中看到一個(gè)類似的算法,它是把原數(shù)組構(gòu)建成一個(gè)大頂堆,然后把大頂堆的第一個(gè)元素與最后一個(gè)元素交換;再把前n-1個(gè)元素重新構(gòu)造成一個(gè)大頂堆,把新大頂堆的第一個(gè)元素與最后一個(gè)元素交換;依此類推,直到新大頂堆只有一個(gè)元素,這樣就得到了一個(gè)有序的二叉堆。
算法如下:
template <class T>
void HeapSort(T a[], int len)
{
      T *ca = new T[len+1];
      ca[0] = 0;
      for (int i=0; i<len; i++)
            ca[i+1] = a[i];

      for (int i=len/2; i>0; i--) //建立初始堆
            HeapAdjust(ca, len, i);

      for (int i=len; i>1; i--)//進(jìn)行l(wèi)en-1次循環(huán),完成堆排序
      {
            Swap(ca[1], ca[i]); //新大頂堆的第一個(gè)元素與最后一個(gè)元素交換
            HeapAdjust(ca, i-1, 1);//篩a[1]元素,得到i-1個(gè)元素的堆
      }

      for (int i=0; i<len; i++)
            a[i] = ca[i+1];

      delete []ca;
}

template <class T>
void HeapAdjust(T a[], int len, int left) //將i與其小兒子交換位置
{
      if (len == 0)
            exit(1);

      T x = a[left];
      int i = left;
      int c = 2 * i;
      while (c <= len)
      {
            if (c < len && a[c+1] > a[c])//若i有右兒子,且右兒子大于左兒子,c指向右兒子
                  c++;
            if (last < a[c])//若i的大兒子大于二叉堆的最后一個(gè)元素,把其移到i的位置
            {
                  a[i] = a[c];
                  i = c;
                  c = 2 * i;
            }
            else
                  break;
      }
      a[i] = x; //把原二叉堆的第一個(gè)元素放到適當(dāng)?shù)目瘴?br>}

template <class T>
void Swap(T & a, T & b)
{
      T t = a;
      a = b;
      b = t;
}

      還有一種方法是每次都要重新調(diào)整大頂堆,使得父親比兒子大,這樣調(diào)整的函數(shù)較簡單,
但因?yàn)槊看味家闅v一半的元素,時(shí)間復(fù)雜度較大。
算法如下:
template <class T>
void HeapSort(T a[], int len)
{
      T *ca = new T[len+1];
      ca[0] = 0;
      for (int i=0; i<len; i++)
            ca[i+1] = a[i];

      for (int i=len/2; i>0; i--) //把原數(shù)組構(gòu)建成一個(gè)大頂堆
            HeapAdjust(ca, len, i);
      Swap(ca[1], ca[len]); //把大頂堆的第一個(gè)元素與最后一個(gè)元素交換
     
      for (int i=len-1; i>0; i--)
      {
            for (int j=i/2; j>0; j--)//遍歷長度為i的堆,得到新的大頂堆
                  HeapAdjust(ca, i, j);
            Swap(ca[1], ca[i]);
      }
     
      for (int i=0; i<len; i++)
            a[i] = ca[i+1];

      delete []ca;
}

template <class T>
void HeapAdjust(T a[], int len, int i) //將i與其小兒子交換位置
{
      int c = 2 * i;

      if (c < len)
      {
            T & max = (a[c] > a[c+1])? a[c] : a[c+1];
            if (a[i] < max)
                  Swap(a[i], max);
      }
      else
      {
            if (a[i] < a[c])
                  Swap(a[i], a[c]);
      }
}

template <class T>
void Swap(T & a, T & b)
{
      T t = a;
      a = b;
      b = t;
}

      模仿構(gòu)造大頂堆的方法,我們可以調(diào)用HeapAdjust()構(gòu)造一個(gè)二叉堆,并提取二叉堆的根到新數(shù)組,
然后把原二叉堆的最后一個(gè)元素放到根的位置,再調(diào)用HeapAdjust()構(gòu)造一個(gè)新二叉堆,依此類推。
算法如下:
template <class T>
void HeapSort(T a[], int len)
{
      T *ca = new T[len+1];
      ca[0] = 0;
      for (int i=0; i<len; i++)
            ca[i+1] = a[i];

      for (int i=len/2; i>0; i--) //把原數(shù)組構(gòu)建成一個(gè)大頂堆
            HeapAdjust(ca, len, i);
      a[0] = ca[1];
      ca[1] = ca[len]; //把二叉堆的最后一個(gè)元素放到根的位置

      for (int i=len-1; i>0; i--)
      {
            for (int j=i/2; j>0; j--)
                  HeapAdjust(ca, i, j);
            a[len-i] = ca[1];
            ca[1] = ca[i]; //把二叉堆的最后一個(gè)元素放到根的位置
      }

      delete []ca;
}

template <class T>
void HeapAdjust(T a[], int len, int i)
{
      int c = 2 * i;

      if (c < len)
      {
            T & min = (a[c] < a[c+1])? a[c] : a[c+1];
            if (a[i] > min)
                  Swap(a[i], min);
      }
      else
      {
            if (a[i] > a[c])
                  Swap(a[i], a[c]);
      }
}

template <class T>
void Swap(T & a, T & b)
{
      T t = a;
      a = b;
      b = t;
}
      后面兩種方法采用的是遞歸,容易理解,但時(shí)間復(fù)雜度較高,因?yàn)楸惹皟煞N要慢上很多,所以不可能是O(nlogn),估計(jì)是O(n^2),但具體我也不會(huì)算,請(qǐng)高手指教。

Posted on 2006-06-14 10:18 夢想飛揚(yáng) 閱讀(4277) 評(píng)論(2)  編輯 收藏 引用

Feedback

# re: 我所理解的堆排序算法  回復(fù)  更多評(píng)論   

2006-10-17 01:49 by 李偉
#define HEAP_MAX_LEN 100
#define DataType int
static DataType heap_table[HEAP_MAX_LEN+1];
static int current_size=0;

void minheap_init(void){
int i;
for(i=0;i<1+HEAP_MAX_LEN;++i)
heap_table[i]=0;
current_size = 0 ;
}

void minheap_filterdown(int start ,int end){
int i= start;//i and father while j as child
int j= i*2+1;
DataType temp= heap_table[i];
while(j<=end)
{
if (heap_table[j]>heap_table[j+1])j++;
if (temp<=heap_table[j])break;
else {
heap_table[i]=heap_table[j];i=j;j=2*j+1;
}
}
heap_table[i]=temp;
}
/*
void swap(DataType *a,DataType *b){
DataType t;
t=*a;*a=*b;*b=t;
}
*/
void minheap_filterup(int start){
int j=start;//i as child while j and father
int i=(j-1)/2;
DataType temp=heap_table[j];
while(j>0){
if (heap_table[i]<=temp)break;
else {heap_table[j]=heap_table[i];j=i;i=(i-1)/2;}
heap_table[j]=temp;
}
}

DataType minheap_fetch(void ){
DataType ret;
if (current_size==0){printf("heap is now empty!\n");}
ret=heap_table[0];heap_table[0]=heap_table[current_size-1];
current_size--;
minheap_filterdown(0,current_size-1);
return ret;
}

int minheap_insert(DataType x){
if (current_size>=HEAP_MAX_LEN){printf("heap is now full!\n");return 0 ;}
heap_table[current_size]= x;
minheap_filterup(current_size);
current_size++;
return 1;
}
int fill[]={8,9,10,7,6,5,2,3,4,1,29,45,67,890};
int main(){
int i=0;
int cnt = sizeof (fill)/4;
minheap_init();
while(i!=cnt){
if (minheap_insert(fill[i])==0)break;
++i;
}
i= 0;
while(i!=cnt){// printf("%d get one from minheap==>%d\n",current_size,heap_table[i]);

printf("get one from minheap==>%d\n",minheap_fetch());
++i;
}
getchar();getchar();getchar();
}

# re: 我所理解的堆排序算法  回復(fù)  更多評(píng)論   

2007-03-31 08:55 by OK
@李偉
你這寫的沒有什么重用性!

只有注冊(cè)用戶登錄后才能發(fā)表評(píng)論。
網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            国产精品一区久久| 欧美在线观看视频在线| 久久精品国产999大香线蕉| 亚洲免费伊人电影在线观看av| 亚洲日本成人女熟在线观看| 激情成人亚洲| 亚洲第一福利社区| 99re热精品| 亚洲在线播放| 久久亚洲精品一区二区| 欧美黄色影院| 一区二区三区四区五区在线| 欧美一进一出视频| 欧美激情亚洲| 国产乱码精品1区2区3区| 午夜精品久久久久99热蜜桃导演| 香港成人在线视频| 欧美.com| 国产精品久久777777毛茸茸| 国内精品久久久| 一区二区免费在线观看| 久久精品亚洲一区二区| 亚洲欧洲精品成人久久奇米网 | 欧美激情一区二区三区四区| 中日韩男男gay无套| 蜜臀av一级做a爰片久久 | 久久精品国产免费| 欧美日韩三级电影在线| 尤妮丝一区二区裸体视频| 亚洲一品av免费观看| 欧美刺激性大交免费视频| 亚洲欧美变态国产另类| 欧美日韩国产在线看| 亚洲国语精品自产拍在线观看| 欧美在线电影| 亚洲综合欧美| 欧美日韩综合网| 日韩香蕉视频| 亚洲激情电影中文字幕| 久久午夜羞羞影院免费观看| 国产日韩一区二区三区| 亚洲欧美综合一区| 亚洲最黄网站| 欧美日韩亚洲综合| 一本久久综合亚洲鲁鲁五月天| 欧美成人视屏| 可以免费看不卡的av网站| 激情综合激情| 久久久久久有精品国产| 亚洲在线不卡| 欧美三区美女| 亚洲一区二区三区免费观看| 亚洲激情一区二区三区| 免费在线成人| 99在线精品观看| 日韩视频一区二区在线观看| 欧美日韩专区在线| 亚洲在线免费视频| 亚洲欧美日韩国产综合精品二区 | 久久午夜电影| 在线看片欧美| 亚洲第一偷拍| 欧美美女操人视频| 亚洲一级片在线观看| 亚洲视频狠狠| 国产欧美日韩三级| 久久色在线观看| 老色鬼精品视频在线观看播放| 伊人久久亚洲影院| 亚洲欧洲精品一区二区精品久久久| 欧美大片专区| 欧美二区在线看| 亚洲毛片在线观看.| 亚洲狼人综合| 国产精品爽爽ⅴa在线观看| 欧美一区二区日韩| 久久乐国产精品| 亚洲精品社区| 亚洲欧美激情一区二区| 黄色成人小视频| 亚洲激情一区| 国产欧美在线观看一区| 久久综合色婷婷| 欧美极品欧美精品欧美视频| 亚洲欧美卡通另类91av| 久久大香伊蕉在人线观看热2| 亚洲电影在线看| 在线综合亚洲| …久久精品99久久香蕉国产| 日韩一区二区精品葵司在线| 国内精品久久久久影院色| 亚洲三级电影全部在线观看高清| 国产精品国产馆在线真实露脸| 久久免费视频观看| 欧美日韩国产首页| 久久综合色播五月| 国产精品av免费在线观看| 蜜臀av国产精品久久久久| 欧美午夜久久久| 欧美成年视频| 国产欧美精品日韩精品| 91久久久在线| 一区二区三区自拍| 亚洲在线视频| 一区二区三区久久久| 久久夜色精品亚洲噜噜国产mv| 亚洲一区日韩在线| 欧美aⅴ一区二区三区视频| 欧美一区二区三区在线看| 欧美成人网在线| 另类激情亚洲| 国产亚洲精品久久飘花| 亚洲最新视频在线| 亚洲毛片一区| 欧美va日韩va| 奶水喷射视频一区| 国产婷婷一区二区| 在线亚洲美日韩| 亚洲精品影院在线观看| 久久久久88色偷偷免费| 久久国产精品久久久久久| 欧美天天影院| 一区二区高清视频| 一卡二卡3卡四卡高清精品视频 | 精品999在线播放| 亚洲欧美色婷婷| 午夜国产不卡在线观看视频| 欧美日韩黄视频| 日韩一区二区精品视频| 日韩视频在线永久播放| 久色婷婷小香蕉久久| 女人色偷偷aa久久天堂| 今天的高清视频免费播放成人 | 麻豆亚洲精品| 麻豆精品传媒视频| 久久精品国产精品| 久久久久久久久久久久久9999| 国产日韩欧美制服另类| 午夜精品久久久久久| 欧美怡红院视频一区二区三区| 国产精品久线观看视频| 亚洲欧美国产高清va在线播| 欧美有码视频| 一区在线电影| 欧美激情一区二区在线| 亚洲另类春色国产| 亚洲男女自偷自拍| 国产手机视频精品| 久久久久国产精品人| 久久婷婷麻豆| 亚洲第一视频网站| 欧美精品黄色| 亚洲五月婷婷| 葵司免费一区二区三区四区五区| 亚洲第一精品夜夜躁人人爽| 欧美激情在线观看| 一本一道久久综合狠狠老精东影业| 亚洲欧美变态国产另类| 国语自产精品视频在线看一大j8 | 欧美激情区在线播放| 一本一道久久综合狠狠老精东影业 | 在线综合视频| 国产欧美日韩91| 久久一二三四| 99re在线精品| 久久偷窥视频| 亚洲一级电影| 亚洲国产精品电影在线观看| 欧美日韩综合另类| 久久婷婷丁香| 亚洲一区三区在线观看| 亚洲第一区色| 久久久综合视频| 亚洲天堂成人在线视频| 国色天香一区二区| 欧美午夜精品电影| 狼人社综合社区| 亚洲欧美日韩视频一区| 亚洲国产欧美一区二区三区丁香婷| 亚洲欧美www| 99国产精品久久久久久久| 国内在线观看一区二区三区| 欧美日韩免费一区二区三区视频 | 国产日韩精品一区二区| 欧美精品一区二区三区很污很色的 | 亚洲私人影吧| 欧美福利视频网站| 午夜影视日本亚洲欧洲精品| 亚洲国产欧美日韩另类综合| 久久视频一区二区| 午夜视频一区| 亚洲无线一线二线三线区别av| 在线欧美不卡| 国内伊人久久久久久网站视频 | 国产一区二区高清视频| 欧美三级电影精品| 欧美日韩免费看| 欧美日韩成人综合| 欧美精品激情| 欧美精品一区视频|