C语言 数据结构堆排序顺序存储(升序)

所属分类: 软件编程 / C 语言 阅读数: 61
收藏 0 赞 0 分享

堆排序顺序存储(升序)

一: 完全二叉树的概念:前h-1层为满二叉树,最后一层连续缺失右结点!

二:首先堆是一棵全完二叉树:

a:构建一个堆分为两步:⑴创建一棵完全二叉树      ⑵调整为一个堆

(标注:大根堆为升序,小根堆为降序)

   b:算法描述:①创建一棵完全二叉树  

②while(有双亲){
A:调整为大根堆;
B:交换根和叶子结点;
C:砍掉叶子结点;
}

  c:时间复杂度为 O(nlogn)  ,空间复杂度为 O(1), 是不稳定排序!

代码实现:

/*堆排序思想:[完全二叉树的定义:前 h-1 层为满二叉树一最后一层连续缺失右结点(即右子女)],(大根堆升序排序,小根堆降序排列) 
  首先堆是一个完全二叉树 ,根据数组下标就可建成了一棵完全二叉树 
  其次:while(有双亲){ 
    A: 调整为一个大根堆         【Adjust()函数实现】 
    B: 交换最后一个叶子结点和根结点    【Swap()函数实现】 
    C: 砍掉最后一个叶子结点      【即元素个数 n--】 
  } 
*/ 
 
#include <iostream> 
#define N 100 
 
using namespace std;  
 
int b[N]={0};    //存储数据的数组  
int n=0;      //记录数据的总个数【0单元不要,实际元素个数为(n-1)个】 
 
void Swap(int *x,int *y){ 
  int t; 
  t=*x; 
  *x=*y; 
  *y=t; 
}  
 
void Adjust(){ 
  int p;         //记录双亲结点  
  int tag=1;       //记录是否已经调整为大根堆(标志性的变量) 
  while(tag){       //判断是否已经调整好为大根堆 
    p=(n-1)/2;     //最后一个双亲结点的下标 
    tag=0;       //凡是交换后,tag=1,标志着还没有调整为大根堆,否则继续调整  
    while(p>0){     //确保有双亲结点 
      if(b[p]<b[2*p]){     //若根结点大于左子女结点,就交换  
        Swap(&b[p],&b[2*p]); 
        tag=1; 
      } 
      if(2*p+1<n && b[p]<b[2*p+1]){ //若存在右子女,并且根结点大于右子女结点,就交换  
        Swap(&b[p],&b[2*p+1]); 
        tag=1;      
      } 
      p--;        //直到最后一个双亲结点调整完  
    }  
  }  
} 
 
void HeapSort(){ 
  while(n>2){         //保证有双亲结点  
    Adjust();        //调整大根堆函数 
    Swap(&b[1],&b[n-1]);  //将最后一个叶子结点和根结点交换  
    n--;          //裁剪最后的叶子结点  
  } 
}   
    
int main(void){ 
  int i,m; 
  cout<<"请输入数据的总数【0单元不要,实际元素个数为(n-1)个】:"<<endl; 
  cin>>n; 
  m=n; 
  cout<<"请输入各个数据【0单元不要,实际元素个数为(n-1)个】:"<<endl; 
  b[0]=0; 
  for(i=1;i<n;i++){ 
    cin>>b[i]; 
  } 
  HeapSort();           //堆排序 
  cout<<"大根堆升序排列为:"<<endl; 
  for(i=1;i<m;i++){ 
    cout<<b[i]<<" "; 
  }  
  cout<<endl; 
  return 0; 
} 

感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!

更多精彩内容其他人还在看

C++中四种对象生存期和作用域以及static的用法总结分析

以下是对C++中四种对象生存期和作用域以及static的用法进行了详细的介绍,需要的朋友可以过来参考下
收藏 0 赞 0 分享

C++嵌套类与局部类详细解析

从作用域的角度看,嵌套类被隐藏在外围类之中,该类名只能在外围类中使用。如果在外围类之外的作用域使用该类名时,需要加名字限定
收藏 0 赞 0 分享

C++空类详解

以下是对C++中的空类进行了详细的介绍,需要的朋友可以过来参考下
收藏 0 赞 0 分享

C++之友元:友元函数和友元类详解

友元是一种允许非类成员函数访问类的非公有成员的一种机制。可以把一个函数指定为类的友元,也可以把整个类指定为另一个类的友元
收藏 0 赞 0 分享

C++中返回指向函数的指针示例

int (*ff(int)) (int *,int);表示:ff(int)是一个函数,带有一个int型的形参,该函数返回int (*) (int *,int),它是一个指向函数的指针,所指向的函数返回int型并带有两个分别是Int*和int型的形参
收藏 0 赞 0 分享

C数据结构之单链表详细示例分析

以下是对C语言中的单链表进行了详细的分析介绍,需要的朋友可以过来参考下
收藏 0 赞 0 分享

C数据结构之双链表详细示例分析

以下是对c语言中的双链表进行了详细的分析介绍,需要的朋友可以过来参考下
收藏 0 赞 0 分享

浅析如何在c语言中调用Linux脚本

如何在c语言中调用Linux脚本呢?下面小编就为大家详细的介绍一下吧!需要的朋友可以过来参考下
收藏 0 赞 0 分享

深入解析unsigned int 和 int

以下是对unsigned int和int进行了详细的分析介绍,需要的朋友可以过来参考下
收藏 0 赞 0 分享

浅谈C++中的string 类型占几个字节

本篇文章小编并不是为大家讲解string类型的用法,而是讲解我个人比较好奇的问题,就是string 类型占几个字节
收藏 0 赞 0 分享
查看更多