数据结构之堆底层实现的循序渐进

题外话

把没写的都补回来!

正题

概念

堆是一棵完全二叉树,因此可以层序的规则采用顺序的方式来高效存储,

大根堆:指根结点比左右孩子都大的堆

小根堆:指根结点比左右孩子都小的堆

性质

1.堆中某个节点的值总是不大于或不小于其父节点的值

2.堆总是一棵完全二叉树。

向下调整

将数组调整为大根堆:

1.从最后一颗子树开始调整,利用二叉树的性质找到最后一颗树的父亲节点 (i-1)/2

2.获取左右孩子的最大值和根节点比较,如果比根节点大就交换,并且继续向下调整

3.调整完结点之后,让结点下标减1,即可从下往上调整为大根堆,一直调整到0下标

底层代码实现及详解

复制代码
public class HeapBottom {
//创建数组
    private int[] elem;
//计数数组元素个数
    public int usedSize;
    public HeapBottom()
    {
        this.elem=new int[10];
    }
    //初始化elem数组
     public void intElem(int[] array)
     {
         for (int i = 0; i < array.length; i++) {
             elem[i]=array[i];
             usedSize++;
         }
     }
    //创建大根堆
     public void createHeap()
     {

         for (int parent = (elem.length-1)/2; parent>=0 ; parent--) {
//向下调整
        siftDown(parent,usedSize);

         }
     }
     //向下调整
    private void siftDown(int parent,int len)
    {
//创建child为parent左子树
        int child=parent*2+1;
//child小于数组长度时
        while (child<len)
        {
//如果child+1也小于数组长度,并且左子树小于右子树
            if (child+1<len&&elem[child]<elem[child+1])
            {
//将右子树下标赋给左子树,此时child相当于右子树
                child=child+1;
            }
//如果右子树大于父亲结点
            if (elem[child]>elem[parent])
            {
//交换元素值
                
复制代码
                swap(parent,child);
复制代码
//并继续向下调整(因为刚交换完的结点不一定是交换后的最大值),将交换完的子树下标给到parent
                parent=child;
//让child为parent左孩子
                child=parent*2+1;
            }
//如果右子树不大于父亲节点,则退出
            else {
                break;
            }

        }

    }

//交换代码

private void swap(int i,int j)

{

int temp=elem[i];

elem[i]=elem[j];

elem[j]=temp;

}

}

添加元素

//添加元素,是在最后一个位置添加元素,然后再向上调整

public void push(int val)

{

//判断是否满了,满了扩容

if (isFull())

{

Arrays.copyOf(elem,elem.length*2);

}

//赋值给最后一个元素

elem[usedSize]=val;

//向上调整

siftUp(usedSize);

usedSize++;

}public boolean isFull()

{

return usedSize==elem.length;

}

//向上调整

public void siftUp(int child)

{

int parent=(child-1)/2;

while (child>0) {

if (elem[child] > elem[parent]) {

swap(child, parent);

child = parent;

parent = (child - 1) / 2;

}

else {

break;

}

}

}

删除元素

//删除元素(数组0下标位置元素和最后一个元素互换位置,然后向下调整即可)

public int pop() {

//判断是否为空数组,空了返回-1(自定义异常也可以)

if (empty())

{

return -1;

}

//将删除元素记录下来

int tmp=elem[0];

//互换位置

swap(0,usedSize-1);

//数组元素数量-1

usedSize--;

// 向下调整

siftDown(0,usedSize);

return tmp;

}

//判断是否为空

public boolean empty() {

return usedSize==0;

}

}

小结

今天还没吃饭,先去吃饭休息休息!!

相关推荐
sp_fyf_2024几秒前
计算机前沿技术-人工智能算法-大语言模型-最新研究进展-2024-10-03
人工智能·算法·机器学习·计算机视觉·语言模型·自然语言处理
陈逸轩*^_^*8 分钟前
Java 网络编程基础
java·网络·计算机网络
这孩子叫逆20 分钟前
Spring Boot项目的创建与使用
java·spring boot·后端
星星法术嗲人24 分钟前
【Java】—— 集合框架:Collections工具类的使用
java·开发语言
Ljubim.te29 分钟前
软件设计师——数据结构
数据结构·笔记
Eric.Lee202136 分钟前
数据集-目标检测系列- 螃蟹 检测数据集 crab >> DataBall
python·深度学习·算法·目标检测·计算机视觉·数据集·螃蟹检测
一丝晨光42 分钟前
C++、Ruby和JavaScript
java·开发语言·javascript·c++·python·c·ruby
天上掉下来个程小白44 分钟前
Stream流的中间方法
java·开发语言·windows
林辞忧1 小时前
算法修炼之路之滑动窗口
算法
xujinwei_gingko1 小时前
JAVA基础面试题汇总(持续更新)
java·开发语言