Java语言实现二叉堆的打印代码分享

作者:GoldArowana 时间:2021-11-27 23:00:15 

二叉堆是一种特殊的堆,二叉堆是完全二元树(二叉树)或者是近似完全二元树(二叉树)。二叉堆有两种:最大堆和最小堆。最大堆:父结点的键值总是大于或等于任何一个子节点的键值;最小堆:父结点的键值总是小于或等于任何一个子节点的键值。

打印二叉堆:利用层级关系

Java语言实现二叉堆的打印代码分享

我这里是先将堆排序,然后在sort里执行了打印堆的方法printAsTree()


public class MaxHeap<T extends Comparable<? super T>> {
 private T[] data;
 private int size;
 private int capacity;

public MaxHeap(int capacity) {
   this.capacity = capacity;
   this.size = 0;
   this.data = (T[]) new Comparable[capacity + 1];
 }

public MaxHeap(T[] arr) {//heapify,数组建堆
   capacity = arr.length;
   data = (T[]) new Comparable[capacity + 1];
   System.arraycopy(arr, 0, data, 1, arr.length);
   size = arr.length;
   for (int i = size / 2; i >= 1; i--) {
     shiftDown(i);
   }
 }

public int size() {
   return this.size;
 }

public int getCapacity() {
   return this.capacity;
 }

public boolean isEmpty() {
   return size == 0;
 }

public T seekMax() {
   return data[1];
 }

public void swap(int i, int j) {
   if (i != j) {
     T temp = data[i];
     data[i] = data[j];
     data[j] = temp;
   }
 }

public void insert(T item) {
   size++;
   data[size] = item;
   shiftUp(size);
 }

public T popMax() {
   swap(1, size--);
   shiftDown(1);
   return data[size + 1];
 }

public void shiftUp(int child) {
   while (child > 1 && data[child].compareTo(data[child / 2]) > 0) {
     swap(child, child / 2);
     child /= 2;
   }
 }

/**
  * @param a data数组中某个元素的下角标
  * @param b data数组中某个元素的下角标
  * @return 哪个元素大就返回哪个的下角标
  */
 private int max(int a, int b) {
   if (data[a].compareTo(data[b]) < 0) {//如果data[b]大
     return b;//返回b
   } else {//如果data[a]大
     return a;//返回a
   }
 }

/**
  * @param a data数组中某个元素的下角标
  * @param b data数组中某个元素的下角标
  * @param c data数组中某个元素的下角标
  * @return 哪个元素大就返回哪个的下角标
  */
 private int max(int a, int b, int c) {
   int biggest = max(a, b);
   biggest = max(biggest, c);
   return biggest;
 }

public void shiftDown(int father) {
   while (true) {
     int lchild = father * 2;
     int rchild = father * 2 + 1;
     int newFather = father;//这里赋不赋值无所谓,如果把下面这个return改成break,那就必须赋值了

if (lchild > size) {//如果没有左、右孩子
       return;
     } else if (rchild > size) {//如果没有右孩子
       newFather = max(father, lchild);
     } else {//如果有左、右孩子
       newFather = max(father, lchild, rchild);
     }

if (newFather == father) {//如果原父结点就是三者最大,则不用继续整理堆了
       return;
     } else {//父节点不是最大,则把大的孩子交换上来,然后继续往下堆调整,直到满足大根堆为止
       swap(newFather, father);
       father = newFather;//相当于继续shiftDown(newFather)。假如newFather原来是father的左孩子,那就相当于shiftDown(2*father)
     }
   }
 }

public static <T extends Comparable<? super T>> void sort(T[] arr) {
   int len = arr.length;
   MaxHeap<T> maxHeap = new MaxHeap<>(arr);
   maxHeap.printAsTree();
   for (int i = len - 1; i >= 0; i--) {
     arr[i] = maxHeap.popMax();
   }
 }

public static void printArr(Object[] arr) {
   for (Object o : arr) {
     System.out.print(o);
     System.out.print("\t");
   }
   System.out.println();
 }

public void printSpace(int n) {//打印n个空格(在这里用‘\t'来代替)
   for (int i = 0; i < n; i++) {
     System.out.printf("%3s", "");
   }
 }

public void printAsTree() {
   int lineNum = 1;//首先遍历第一行
   int lines = (int) (Math.log(size) / Math.log(2)) + 1;//lines是堆的层数
   int spaceNum = (int) (Math.pow(2, lines) - 1);
   for (int i = 1; i <= size; ) { //因为在[1...size]左闭右闭区间存数据,data[0]不存数据

//每层都是打印这个区间[2^(层数-1) ... (2^层数)-1]。如果堆里的数不够(2^层数)-1个,那就打印到size。所以取min((2^层数)-1,size).
     for (int j = (int) Math.pow(2, lineNum - 1); j <= Math.min(size, (int) Math.pow(2, lineNum) - 1); j++) {
       printSpace(spaceNum); //打印spaceNum个空格
       System.out.printf("%3s", data[j]);//打印数据
       System.out.printf("%3s", "");//图片中绿色方框
       printSpace(spaceNum);//打印spaceNum个空格
       i++;//每打印一个元素就 + 1
     }
     lineNum++;
     spaceNum = spaceNum / 2;
     System.out.println();
   }
 }

public static void main(String args[]) {
   Integer[] arr = {3, 5, 1, 7, 2, 9, 8, 0, 4, 6, 1, 3, 6, 1, 1};
   sort(arr);
 }
}

执行结果:

Java语言实现二叉堆的打印代码分享

来源:http://www.cnblogs.com/noKing/p/7966272.html

标签:java,二叉
0
投稿

猜你喜欢

  • SpringBoot整合Quartz实现定时任务详解

    2021-11-22 06:40:57
  • 逐步讲解快速排序算法及C#版的实现示例

    2022-09-09 14:36:19
  • SpringIOC容器Bean的作用域及生命周期实例

    2023-10-01 11:21:34
  • Spring boot热部署devtools过程解析

    2022-06-21 13:12:09
  • Mybatis对mapper的加载流程深入讲解

    2022-06-01 12:33:04
  • java Hibernate多对多映射详解及实例代码

    2023-07-02 07:24:40
  • Unity游戏开发实现场景切换示例

    2022-12-12 16:06:20
  • java实现二叉树的创建及5种遍历方法(总结)

    2022-03-14 09:00:28
  • Java实现驼峰和下划线互相转换的示例代码

    2023-12-16 15:23:48
  • java实现酒店管理系统

    2023-06-09 23:59:50
  • SpringBoot如何实现定时任务示例详解

    2023-10-11 23:24:42
  • Android实现界面跳转功能

    2022-05-07 21:51:32
  • 详解Java动态字节码技术

    2022-06-20 03:20:20
  • 基于AForge实现C#摄像头视频录制功能

    2023-08-25 02:21:26
  • 使用设计模式中的工厂方法模式进行C#编程的示例讲解

    2023-10-21 05:27:08
  • Java多线程编程之访问共享对象和数据的方法

    2022-10-01 07:05:02
  • Android中BaseActivity自定义标题栏

    2022-04-20 06:47:37
  • 21天学习android开发教程之SurfaceView

    2023-04-17 17:01:56
  • C语言数据结构实现银行模拟

    2023-04-16 17:25:49
  • JDK8中新增的原子性操作类LongAdder详解

    2023-06-19 22:02:58
  • asp之家 软件编程 m.aspxhome.com