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

堆排序顺序存储(升序)

一: 完全二叉树的概念:前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;
}

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

(0)

相关推荐

  • C语言实现数据结构串(堆分配存储表示法)实例详解

    堆分配存储表示法 存储结构: 构建堆来存储字符串,本质上是顺序表 实现代码: #include <stdio.h> #include <stdlib.h> #include <string.h> #define OK 1 #define ERROR 0 #define TRUE 1 #define FALSE 0 #define OVERFLOW -2 #define STR_INIT_SIZE 100 #define STRINCREMENT 10 typedef i

  • C语言中字符串的存储方法

    众所周知,C语言中没有数据类型能够存储字符串,char数据类型仅仅能够存储一个字符的数据,那么在C语言中关于存储字符串这一难题我们改何去何从呢? 下面将详述相关的字符串存储方法; 1.使用字符数组存; [root@Qrui ruiy]# #include<stdio.h> int main(int argc,const char *argv[],const char **env[]) { char name[] = "qinrui";//定义一个字符数组,并初始化; cha

  • C语言 存储类详解及示例代码

    C 存储类 存储类定义 C 程序中变量/函数的范围(可见性)和生命周期.这些说明符放置在它们所修饰的类型之前.下面列出 C 程序中可用的存储类: auto register static extern auto 存储类 auto 存储类是所有局部变量默认的存储类. { int mount; auto int month; } 上面的实例定义了两个带有相同存储类的变量,auto 只能用在函数内,即 auto 只能修饰局部变量. register 存储类 register 存储类用于定义存储在寄存器

  • C语言线性表顺序存储结构实例详解

    C语言线性表顺序存储结构实例详解 1. 什么是顺序存储结构? 用一段地址连续的存储单元依次存储线性表的数据元素. 2.线性表的顺序存储结构 #include<stdio.h> #include<stdlib.h> #define Max 80 //存储空间初始分配量 #define Increment 10 //存储空间分配增量 typedef struct { int *elem; // 存储空间基地址,此处为int型,视情况而定 int length; // 元素表当前长度 i

  • C语言基础知识变量的作用域和存储方式详细介绍

    变量的作用域和存储方式 1.简述变量按作用域的分类 变量按作用域分:分为全局变量和局部变量 全局变量:在所有函数外部定义的变量叫做全局变量 全局变量的使用范围:从定义位置开始到下面整个程序结束 局部变量:在一个函数内部定义的变量或者函数的形式参数统称为局部变量 局部变量的使用范围:在函数内部定义的变量只能在本函数内部进行使用 2.简述变量按存储方式的分类 静态变量 自动变量 寄存器变量[寄存器就是cpu内部可以存储数据的一些硬件东西] 3.简述全局变量和局部变量命名冲突的问题 1>在一个函数内部

  • IOS开发之路--C语言存储方式和作用域

    概述 基本上每种语言都要讨论这个话题,C语言也不例外,因为只有你完全了解每个变量或函数存储方式.作用范围和销毁时间才可能正确的使用这门语言.今天将着重介绍C语言中变量作用范围.存储方式.生命周期.作用域和可访问性. 变量作用范围 存储方式 可访问性 变量作用范围 在C语言中变量从作用范围包括全局变量和局部变量.全局变量在定义之后所有的函数中均可以使用,只要前面的代码修改了,那么后面的代码中再使用就是修改后的值:局部变量的作用范围一般在一个函数内部(通常在一对大括号{}内),外面的程序无法访问它,

  • iOS开发系列--详细讲解C语言之存储方式和作用域

    概述 基本上每种语言都要讨论这个话题,C语言也不例外,因为只有你完全了解每个变量或函数存储方式.作用范围和销毁时间才可能正确的使用这门语言.今天将着重介绍C语言中变量作用范围.存储方式.生命周期.作用域和可访问性. 变量作用范围 存储方式 可访问性 变量作用范围 在C语言中变量从作用范围包括全局变量和局部变量.全局变量在定义之后所有的函数中均可以使用,只要前面的代码修改了,那么后面的代码中再使用就是修改后的值:局部变量的作用范围一般在一个函数内部(通常在一对大括号{}内),外面的程序无法访问它,

  • C语言二进制思想以及数据的存储

    我们平时使用的数字都是由 0~9 共十个数字组成的,例如 1.9.10.297.952 等,一个数字最多能表示九,如果要表示十. 十一.二十九.一百等,就需要多个数字组合起来. 例如表示 5+8 的结果,一个数字不够,只能"进位",用 13 来表示:这时"进一位"相当于十,"进两位"相当于二十. 因为逢十进一,也因为只有 0~9 共十个数字,所以叫做十进制(Decimalism). 进制也就是进位制.在进行加法(减法)运算时,逢X进(借)一就是X

  • C语言 数据结构之连续存储数组的算法

    数据结构之数组定义及基本操作 数据结构中最基本的一个结构就是线性结构,而线性结构又分为连续存储结构和离散存储结构.所谓的连续存储结构其实就是数组. 数组本质其实也是数据的一种存储方式,既然有了数据的存储,就会涉及到如何对数据进行寻址的问题.首先,先说一下在数组中数据是如何存储的,在内存中,数组中的数据是以一组连续的数据集合的形式存在于内存中.当我们访问存在于内存中的数组时,我们应该找到其在内存中的地址,当我们找到数据的地址后我们就可以找到对应的数据.了解了以上知识后,我们就可以进行数组的设计了(

  • C语言 二叉树的链式存储实例

    二叉树的链式存储 实现二叉树的基本操作:建立.遍历.计算深度.结点数.叶子数等. 输入C,先序创建二叉树,#表示空节点: 输入H:计算二叉树的高度: 输入L:计算二叉树的叶子个数: 输入N:计算二叉树节点总个数: 输入1:先序遍历二叉树: 输入2:中序遍历二叉树: 输入3:后续遍历二叉树: 输入F:查找值=x的节点的个数: 输入P:以缩格文本形式输出所有节点. 很简单就不需要多解释了,代码贴上 #include <stdio.h> #include <stdlib.h> #incl

随机推荐