初识简单 堆

概念

是一种特殊的数据结构,它可以被视为一棵完全二叉树的数组表示

按完全二叉树的顺序存储方式存储 在一 个一维数组中

实际就是在完全二叉树的基础上进行了一些调整

性质

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

总是一棵完全二叉树。

此外 堆的构建对于非完全二叉树,则不适合使用顺序方式进行存储,存入在数组中必须要存储空节 点,就会导致空间利用率比较低。

构建

构建堆分为向下调整 和 向上调整

向下调整

复制代码
public void createHeap(){ 
    for(int parent = (usedSize - 1 -1)/ 2 ; parent > 0;parent--){                    呃  siftDown(parent,usedSize);   
} 
}
private void siftDown(int parent, int usedSize) {
   int child = 2 * parent - 1;
   while (child < usedSize){
      if (child + 1 < usedSize && elem[child] < elem[child+1]) {
                child++;
      }
      if (elem[child] > elem[child+1]) {
         swap(elem,child,parent);
      }
   }
}

向上调整(时间复杂度相对向下调整高)

复制代码
private  void siftUp(int child){
 int parent = (child -1) / 2;
 while(parent >= 0 ){
    if(elem[parent] < elem [child]){
       swap(elem,child,parent);
       child = parent;
       parent = (child-1) / 2;
    }
    else{
       break;
    }
 }
}

时间复杂度

堆的插入与删除

在完成堆的插入与删除时要考虑当数组为 和为的情况

复制代码
public boolean isFull(){
   return  this.usedSize == elem.length;
}
public boolean isEmpty() {
   return usedSize == 0;
}

offer

//已满 扩容数组

复制代码
if(isFull()){
   elem = Arrays.copyOf(elem,2*elem.length);
}
//拿到值 usedSize向后移动一步
elem[usedSize] =val;
usedSize++;

poll

//删除 从栈顶元素删除

复制代码
if(isEmpty()){
   return  -1;
}
int val = elem[0];
swap(elem,0,usedSize - 1 );
siftDown(0,usedSize -1 );
usedSize--;
return  val;

PriorityQueue 优先级队列

PriorityQueue底层使用了这种数据结构

Java集合框架中提供了PriorityQueuePriorityBlockingQueue 两种类型的优先级队列PriorityQueue 是线程 不安全的

PriorityBlockingQueue 是线程 安全的

因此堆中的方法都是适用于PriorityQueue

boolean offer (E e) E peek () E poll () int size () void clear () boolean isEmpty()

堆的应用

用堆作为底层结构封装 优先级队列

堆排序

建堆

升序:建大堆

降序:建小堆

堆删除思想来进行排序

复制代码
heapsort
复制代码
public void heapsort() {
   int end = usedSize - 1;
   while (end > 0) {
      swap(elem, 0, end);
      siftDown(0, end);
      end--;
   }
}

Top-k

最小K : 建立大根堆

最大K : 建立小根堆

这里提供两种方法解决

1.整体法

public int[] smallestK(int[] arr, int k) {

PriorityQueue<Integer> priorityqueue = new PriorityQueue<>();

for(int i = 0; i < arr.length; i++){

priorityqueue.offer(arr[i]);

}

int [] ret =new int [k]; for(int i = 0 ; i < k ; i ++){

ret[i] = priorityqueue.poll();

}

return ret;

}

2.分部法

复制代码
 public int[] smallestK(int[] arr, int k) {
        PriorityQueue<Integer> priorityqueue = new PriorityQueue<>();
      for(int  i = 0; i < arr.length; i++){
        priorityqueue.offer(arr[i]);
      }
      //n-k
      for(int i = k;  i < arr.length ; i++ ){
        int peekVal = priorityqueue.peek();
        if(arr[i] < peekVal){
            priorityqueue.poll();
            priorityqueue.offer(arr[i]);
        }
      }
        int[] ret  = new int [k];
        for(int j = 0 ;j < k ; j++){
            ret[j] =  priorityqueue.poll();
        }
        return ret;
     
    }
相关推荐
后端AI实验室33 分钟前
用AI写代码,我差点把漏洞发上线:血泪总结的10个教训
java·ai
CoovallyAIHub41 分钟前
Moonshine:比 Whisper 快 100 倍的端侧语音识别神器,Star 6.6K!
深度学习·算法·计算机视觉
CoovallyAIHub2 小时前
速度暴涨10倍、成本暴降6倍!Mercury 2用扩散取代自回归,重新定义LLM推理速度
深度学习·算法·计算机视觉
CoovallyAIHub2 小时前
实时视觉AI智能体框架来了!Vision Agents 狂揽7K Star,延迟低至30ms,YOLO+Gemini实时联动!
算法·架构·github
CoovallyAIHub2 小时前
开源:YOLO最强对手?D-FINE目标检测与实例分割框架深度解析
人工智能·算法·github
程序员清风2 小时前
小红书二面:Spring Boot的单例模式是如何实现的?
java·后端·面试
belhomme3 小时前
(面试题)Redis实现 IP 维度滑动窗口限流实践
java·面试
CoovallyAIHub3 小时前
OpenClaw:从“19万星标”到“行业封杀”,这只“赛博龙虾”究竟触动了谁的神经?
算法·架构·github
Be_Better3 小时前
学会与虚拟机对话---ASM
java
刀法如飞3 小时前
程序员必须知道的核心算法思想
算法·编程开发·算法思想