热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

PriorityQueue源码分析

 publicbooleanhasNext(){returncursor<size||(forgetMeNot!null&am

PriorityQueue 源码分析

 

        public boolean hasNext() {
            return cursor 
                (forgetMeNot != null && !forgetMeNot.isEmpty());
        }

        //内部的迭代器,就是数组的迭代,迭代指针cursor,
        public E next() {
            if (expectedModCount != modCount)
                throw new ConcurrentModificationException();
            if (cursor < size)
                return (E) queue[lastRet = cursor++];
            if (forgetMeNot != null) {
                lastRet = -1;
                lastRetElt = forgetMeNot.poll();
                if (lastRetElt != null)
                    return lastRetElt;
            }
            throw new NoSuchElementException();
        }

PriorityQueue 源码分析

扩容:
private void grow(int minCapacity) {
        int oldCapacity = queue.length;
        // 小于64扩大为原来2倍,大于等于64扩大为原来1.5倍,
        int newCapacity = oldCapacity + ((oldCapacity <64) ?
                                         (oldCapacity + 2) :
                                         (oldCapacity >> 1));
        // overflow-conscious code
        if (newCapacity - MAX_ARRAY_SIZE > 0)
            newCapacity = hugeCapacity(minCapacity);
        queue = Arrays.copyOf(queue, newCapacity);//丢弃原来的数组,指向新的数组,
    }
public class zusheQueue {
private static PriorityQueue queue = new PriorityQueue(3);


  public static void main(String[] args)  {
   queue.offer(80);
   queue.offer(60);
   queue.offer(70);
   queue.offer(40);
   queue.offer(20);
   queue.offer(10);
   queue.offer(90);
   queue.offer(22);
   queue.offer(15);
   queue.offer(4);
   queue.offer(1);
   System.out.println(queue);//[1, 4, 20, 22, 10, 70, 90, 80, 40, 60, 15]   数组实现最小堆
    System.out.println("出来 "+queue.poll());
    System.out.println("出来 "+queue.poll());
    System.out.println("出来 "+queue.poll());
    System.out.println("出来 "+queue.poll());
    /*出来 1
    出来 4
    出来 10
    出来 15    出来最小的*/
    System.out.println(queue);
    
    
    Iterator i = queue.iterator();
    while(i.hasNext()) {
          System.out.print(i.next() + " ");//1 4 20 22 10 70 90 80 40 60 15 
       }
       
       
    List l = Arrays.asList(7,85,4,9,5,85,74,5,8);
    PriorityQueue p = new PriorityQueue<>(l);
    System.out.println(p);
  }
    
}

PriorityQueue 源码分析

PriorityQueue 源码分析

private void siftUpComparable(int k, E x) {
    Comparablesuper E> key = (Comparablesuper E>) x;
    while (k > 0) {
        int parent = (k - 1) >>> 1;  //无符号右移
        Object e = queue[parent];
        if (key.compareTo((E) e) >= 0)
            break;
        queue[k] = e;
        k = parent;
    }
    queue[k] = key;
}

PriorityQueue 源码分析

PriorityQueue 源码分析

PriorityQueue 源码分析

PriorityQueue 源码分析

PriorityQueue 源码分析

PriorityQueue 源码分析

PriorityQueue 源码分析

private void siftDownComparable(int k, E x) { // k = 0
        Comparablesuper E> key = (Comparablesuper E>)x;
        int half = size >>> 1;        // 非叶子节点
        while (k < half) {
            int child = (k <<1) + 1;   // child = 左孩子
            Object c = queue[child]; // c = 左孩子
            int right = child + 1;//right = 右孩子
            if (right 
                ((Comparablesuper E>) c).compareTo((E) queue[right]) > 0)  //有右孩子,并且左孩子大于右孩子
                c = queue[child = right];// c = 右孩子,child = 右孩子
            if (key.compareTo((E) c) <= 0)  //key小于右孩子,就是小于所有孩子
                break; //结束循环
            queue[k] = c; //  否则key大于右孩子,k位置是右孩子,就是较小的孩子
            k = child;  //指针k值为right右孩子的位置, 比较都是根据指针位置比较,放的时候才放对象。
        }
        queue[k] = key;
    }

PriorityQueue 源码分析

PriorityQueue 源码分析

PriorityQueue 源码分析

构造函数:

PriorityQueue 源码分析

public PriorityQueue(Collectionextends E> c) {
    if (c instanceof SortedSet) {
        SortedSetextends E> ss = (SortedSetextends E>) c;
        this.comparator = (Comparatorsuper E>) ss.comparator();
        initElementsFromCollection(ss);
    }
    else if (c instanceof PriorityQueue) {
        PriorityQueueextends E> pq = (PriorityQueueextends E>) c;
        this.comparator = (Comparatorsuper E>) pq.comparator();
        initFromPriorityQueue(pq);
    }
    else {
        this.comparator = null;
        initFromCollection(c);
    }
}
private void initFromCollection(Collectionextends E> c) {
    initElementsFromCollection(c);
    heapify();
}
private void initElementsFromCollection(Collectionextends E> c) {
    Object[] a = c.toArray();
    // If c.toArray incorrectly doesn't return Object[], copy it.
    if (a.getClass() != Object[].class)
        a = Arrays.copyOf(a, a.length, Object[].class);
    int len = a.length;
    if (len == 1 || this.comparator != null)
        for (int i = 0; i )
            if (a[i] == null)
                throw new NullPointerException();
    this.queue = a;
    this.size = a.length;
}
private void heapify() {
    for (int i = (size >>> 1) - 1; i >= 0; i--)   //(size >>> 1) - 1就是找到第一个非叶子节点,然后从下到上从右到左,
        siftDown(i, (E) queue[i]);
}

PriorityQueue 源码分析

PriorityQueue 源码分析

 

public E poll() {
    if (size == 0)
        return null;
    int s = --size;
    modCount++;
    E result = (E) queue[0];   //取出第0个元素
    E x = (E) queue[s];     //取出最后一个元素
    queue[s] = null;   // 最后一个元素置为空
    if (s != 0)
        siftDown(0, x);   //  最后一个元素不要先不要放在第0个元素位置,比较之后确定位置了再放。放置的都是左右孩子,改变的是k的值,key不到最后不放。
    return result;  //返回第0个元素
}
private void siftDownComparable(int k, E x) {
    Comparablesuper E> key = (Comparablesuper E>)x;    
    int half = size >>> 1;      
    while (k < half) {
        // sI, sV是左右子节点的较小位置和值,k,key是要比较的元素和准备放的位置(落之前要比较在放)
        int sI = (k <<1) + 1; // 较小位置=左孩子索引
        Object sV = queue[sI];//较小值=左孩子
        int right = sI+ 1; 
        if (right 
            ((Comparablesuper E>) sV ).compareTo((E) queue[right]) > 0)   //有右孩子,并且左孩子大于右孩子,
            sV  = queue[sI= right];     //  较小位置=右孩子索引,较小值=右孩子,
        if (key.compareTo((E) sV ) <= 0)    //key比左右都小,放上去,
            break;
        queue[k] = sV ;//否则key比子节点要大,交换位置,并且自己准备放在sl的位置。放置的都是左右孩子,改变的是k的值,key不到最后不放。
        k = sI;  //自己放在sl的位置之前,也要跟子节点比较一下,在放。
    }
    queue[k] = key;//key放在k的位置
}

 


推荐阅读
  • C# 7.0 新特性:基于Tuple的“多”返回值方法
    本文介绍了C# 7.0中基于Tuple的“多”返回值方法的使用。通过对C# 6.0及更早版本的做法进行回顾,提出了问题:如何使一个方法可返回多个返回值。然后详细介绍了C# 7.0中使用Tuple的写法,并给出了示例代码。最后,总结了该新特性的优点。 ... [详细]
  • 阿,里,云,物,联网,net,core,客户端,czgl,aliiotclient, ... [详细]
  • Java太阳系小游戏分析和源码详解
    本文介绍了一个基于Java的太阳系小游戏的分析和源码详解。通过对面向对象的知识的学习和实践,作者实现了太阳系各行星绕太阳转的效果。文章详细介绍了游戏的设计思路和源码结构,包括工具类、常量、图片加载、面板等。通过这个小游戏的制作,读者可以巩固和应用所学的知识,如类的继承、方法的重载与重写、多态和封装等。 ... [详细]
  • Monkey《大话移动——Android与iOS应用测试指南》的预购信息发布啦!
    Monkey《大话移动——Android与iOS应用测试指南》的预购信息已经发布,可以在京东和当当网进行预购。感谢几位大牛给出的书评,并呼吁大家的支持。明天京东的链接也将发布。 ... [详细]
  • Iamtryingtomakeaclassthatwillreadatextfileofnamesintoanarray,thenreturnthatarra ... [详细]
  • 本文分享了一个关于在C#中使用异步代码的问题,作者在控制台中运行时代码正常工作,但在Windows窗体中却无法正常工作。作者尝试搜索局域网上的主机,但在窗体中计数器没有减少。文章提供了相关的代码和解决思路。 ... [详细]
  • 本文介绍了使用Java实现大数乘法的分治算法,包括输入数据的处理、普通大数乘法的结果和Karatsuba大数乘法的结果。通过改变long类型可以适应不同范围的大数乘法计算。 ... [详细]
  • Java序列化对象传给PHP的方法及原理解析
    本文介绍了Java序列化对象传给PHP的方法及原理,包括Java对象传递的方式、序列化的方式、PHP中的序列化用法介绍、Java是否能反序列化PHP的数据、Java序列化的原理以及解决Java序列化中的问题。同时还解释了序列化的概念和作用,以及代码执行序列化所需要的权限。最后指出,序列化会将对象实例的所有字段都进行序列化,使得数据能够被表示为实例的序列化数据,但只有能够解释该格式的代码才能够确定数据的内容。 ... [详细]
  • Java容器中的compareto方法排序原理解析
    本文从源码解析Java容器中的compareto方法的排序原理,讲解了在使用数组存储数据时的限制以及存储效率的问题。同时提到了Redis的五大数据结构和list、set等知识点,回忆了作者大学时代的Java学习经历。文章以作者做的思维导图作为目录,展示了整个讲解过程。 ... [详细]
  • 本文讨论了如何优化解决hdu 1003 java题目的动态规划方法,通过分析加法规则和最大和的性质,提出了一种优化的思路。具体方法是,当从1加到n为负时,即sum(1,n)sum(n,s),可以继续加法计算。同时,还考虑了两种特殊情况:都是负数的情况和有0的情况。最后,通过使用Scanner类来获取输入数据。 ... [详细]
  • 本文介绍了OC学习笔记中的@property和@synthesize,包括属性的定义和合成的使用方法。通过示例代码详细讲解了@property和@synthesize的作用和用法。 ... [详细]
  • JavaSE笔试题-接口、抽象类、多态等问题解答
    本文解答了JavaSE笔试题中关于接口、抽象类、多态等问题。包括Math类的取整数方法、接口是否可继承、抽象类是否可实现接口、抽象类是否可继承具体类、抽象类中是否可以有静态main方法等问题。同时介绍了面向对象的特征,以及Java中实现多态的机制。 ... [详细]
  • Spring特性实现接口多类的动态调用详解
    本文详细介绍了如何使用Spring特性实现接口多类的动态调用。通过对Spring IoC容器的基础类BeanFactory和ApplicationContext的介绍,以及getBeansOfType方法的应用,解决了在实际工作中遇到的接口及多个实现类的问题。同时,文章还提到了SPI使用的不便之处,并介绍了借助ApplicationContext实现需求的方法。阅读本文,你将了解到Spring特性的实现原理和实际应用方式。 ... [详细]
  • 本文讨论了一个关于cuowu类的问题,作者在使用cuowu类时遇到了错误提示和使用AdjustmentListener的问题。文章提供了16个解决方案,并给出了两个可能导致错误的原因。 ... [详细]
  • HDU 2372 El Dorado(DP)的最长上升子序列长度求解方法
    本文介绍了解决HDU 2372 El Dorado问题的一种动态规划方法,通过循环k的方式求解最长上升子序列的长度。具体实现过程包括初始化dp数组、读取数列、计算最长上升子序列长度等步骤。 ... [详细]
author-avatar
chroalist
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有