集合源码解析StackandQueue精讲初中级人员
前言
集合的重要程度很高 如果不能全面了解 就没法完全发挥它的作用 打好地基才能走得远 Stack & Queue概述
Java里有一个叫做 Stack 的类,却没有叫做 Queue 的类( 它是个接口名字 )。当需要使用栈时,Java已不推荐使用Stack,而是推荐使用更高效的ArrayDeque;既然Queue只是一个接口,当需要使用队列时也就首选ArrayDeque了(次选是LinkedList )。Queue
Queue 接口继承自Collection接口 ,除了最基本的Collection的方法之外,它还支持额外的 *insertion*, *extraction*和*inspection*操作 。这里有两组格式,共6个方法,一组是抛出异常的实现;另外一组是返回值的实现(没有则返回null)。
Throws exception
Returns special value
Insert
add(e)
offer(e)
Remove
remove()
poll()
Examine
element()
peek() Deque
Deque 是"double ended queue", 表示双向的队列,英文读作"deck". Deque 继承自 Queue接口,除了支持Queue的方法之外,还支持insert , remove 和examine 操作,由于Deque是双向的,所以可以对队列的头和尾都进行操作,它同时也支持两组格式,一组是抛出异常的实现;另外一组是返回值的实现(没有则返回null)。共12个方法如下:
First Element - Head
Last Element - Tail
Throws exception
Special value
Throws exception
Special value
Insert
addFirst(e)
offerFirst(e)
addLast(e)
offerLast(e)
Remove
removeFirst()
pollFirst()
removeLast()
pollLast()
Examine
getFirst()
peekFirst()
getLast()
peekLast()
当把 Deque 当做FIFO的queue 来使用时,元素是从deque 的尾部添加,从头部进行删除的; 所以deque 的部分方法是和queue 是等同的。具体如下:
Queue Method
Equivalent Deque Method
add(e)
addLast(e)
offer(e)
offerLast(e)
remove()
removeFirst()
poll()
pollFirst()
element()
getFirst()
peek()
peekFirst()
Deque 的含义是"double ended queue",即双端队列 , 它既可以当作栈使用,也可以当作队列使用 。下表列出了 Deque 与 Queue 相对应的接口:
Queue Method
Equivalent Deque Method
说明
add(e)
addLast(e)
向队尾插入元素,失败则抛出异常
offer(e)
offerLast(e)
向队尾插入元素,失败则返回 false
remove()
removeFirst()
获取并删除队首元素,失败则抛出异常
poll()
pollFirst()
获取并删除队首元素,失败则返回 null
element()
getFirst()
获取但不删除队首元素,失败则抛出异常
peek()
peekFirst()
获取但不删除队首元素,失败则返回 null
下表列出了 Deque 与 Stack 对应的接口:
Stack Method
Equivalent Deque Method
说明
push(e)
addFirst(e)
向栈顶插入元素,失败则抛出异常
无
offerFirst(e)
向栈顶插入元素,失败则返回 false
pop()
removeFirst()
获取并删除栈顶元素,失败则抛出异常
无
pollFirst()
获取并删除栈顶元素,失败则返回 null
peek()
getFirst()
获取但不删除栈顶元素,失败则抛出异常
无
peekFirst()
获取但不删除栈顶元素,失败则返回 null
上面两个表共定义了 Deque 的12个接口。添加,删除,取值都有两套接口,它们功能相同,区别是对失败情况的处理不同。 一套接口遇到失败就会抛出异常,另一套遇到失败会返回特殊值( false 或null ) 。除非某种实现对容量有限制,大多数情况下,添加操作是不会失败的。虽然Deque的接口有12个之多,但无非就是对容器的两端进行操作,或添加,或删除,或查看**。明白了这一点讲解起来就会非常简单。
ArrayDeque和LinkedList 是 Deque 的两个通用实现 ,由于官方更推荐使用 AarryDeque 用作栈和队列,这个文章主要整理 ArrayDeque **的具体实现。
从名字可以看出 ArrayDeque 底层通过数组实现,为了满足可以同时在数组两端插入或删除元素的需求,该数组还必须是循环的,即 循环数组(circular array) ,也就是说数组的任何一点都可能被看作起点或者终点。 ArrayDeque 是非线程安全的(not thread-safe),当多个线程同时使用的时候,需要程序员手动同步;另外,该容器不允许放入 null 元素。
上图中我们看到, head 指向首端第一个有效元素,tail 指向尾端第一个可以插入元素的空位。因为是循环数组,所以head 不一定总等于0,tail 也不一定总是比head 大。方法剖析addFirst()
addFirst(E e) 的作用是在Deque的首端插入元素,也就是在head 的前面插入元素,在空间足够且下标没有越界的情况下,只需要将elements[--head] = e 即可。
实际需要考虑: 1.空间是否够用,以及2.下标是否越界的问题。上图中,如果 head 为0 之后接着调用addFirst() ,虽然空余空间还够用,但head 为-1 ,下标越界了。下列代码很好的解决了这两个问题。//addFirst(E e) public void addFirst(E e) { if (e == null)//不允许放入null throw new NullPointerException(); elements[head = (head - 1) & (elements.length - 1)] = e;//2.下标是否越界 if (head == tail)//1.空间是否够用 doubleCapacity();//扩容 }
上述代码我们看到, 空间问题是在插入之后解决的 ,因为 tail 总是指向下一个可插入的空位,也就意味着elements 数组至少有一个空位,所以插入元素的时候不用考虑空间问题。
下标越界的处理解决起来非常简单, head = (head - 1) & (elements.length - 1) 就可以了,这段代码相当于取余,同时解决了head 为负值的情况。因为elements.length 必需是2 的指数倍,elements - 1 就是二进制低位全1 ,跟head - 1 相与之后就起到了取模的作用,如果head - 1 为负数(其实只可能是-1),则相当于对其取相对于elements.length 的补码。
下面再说说扩容函数 doubleCapacity() ,其逻辑是申请一个更大的数组(原数组的两倍),然后将原数组复制过去。过程如下图所示:
图中我们看到,复制分两次进行,第一次复制 head 右边的元素,第二次复制head 左边的元素。//doubleCapacity() private void doubleCapacity() { assert head == tail; int p = head; int n = elements.length; int r = n - p; // head右边元素的个数 int newCapacity = n << 1;//原空间的2倍 if (newCapacity < 0) throw new IllegalStateException("Sorry, deque too big"); Object[] a = new Object[newCapacity]; System.arraycopy(elements, p, a, 0, r);//复制右半部分,对应上图中绿色部分 System.arraycopy(elements, 0, a, r, p);//复制左半部分,对应上图中灰色部分 elements = (E[])a; head = 0; tail = n; } addLast()
addLast(E e) 的作用是在Deque的尾端插入元素,也就是在tail 的位置插入元素,由于tail 总是指向下一个可以插入的空位,因此只需要elements[tail] = e; 即可。插入完成后再检查空间,如果空间已经用光,则调用doubleCapacity() 进行扩容。
public void addLast(E e) { if (e == null)//不允许放入null throw new NullPointerException(); elements[tail] = e;//赋值 if ( (tail = (tail + 1) & (elements.length - 1)) == head)//下标越界处理 doubleCapacity();//扩容 }
下标越界处理方式 addFirt() 中已经讲过,不再赘述。pollFirst()
pollFirst() 的作用是删除并返回Deque首端元素,也即是head 位置处的元素。如果容器不空,只需要直接返回elements[head] 即可,当然还需要处理下标的问题。由于ArrayDeque 中不允许放入null ,当elements[head] == null 时,意味着容器为空。public E pollFirst() { E result = elements[head]; if (result == null)//null值意味着deque为空 return null; elements[h] = null;//let GC work head = (head + 1) & (elements.length - 1);//下标越界处理 return result; } pollLast()
pollLast() 的作用是删除并返回Deque尾端元素,也即是tail 位置前面的那个元素。public E pollLast() { int t = (tail - 1) & (elements.length - 1);//tail的上一个位置是最后一个元素 E result = elements[t]; if (result == null)//null值意味着deque为空 return null; elements[t] = null;//let GC work tail = t; return result; } peekFirst()
peekFirst() 的作用是返回但不删除Deque首端元素,也即是head 位置处的元素,直接返回elements[head] 即可。public E peekFirst() { return elements[head]; // elements[head] is null if deque empty } peekLast()
peekLast() 的作用是返回但不删除Deque尾端元素,也即是tail 位置前面的那个元素。public E peekLast() { return elements[(tail - 1) & (elements.length - 1)]; }
文旅局长卷上天,流量如何变留量?本文收录于广东文旅拼出高质量谁也没想到,从开年至今,各地文旅局长打擂台续集不断。上天下海,穿越古今,实实在在的流量热度不断狂飙,甚至还引来了蹭热搜的冒牌局长杭州文旅局局长高清写真等
万亩油菜笑烂漫禁毒宣讲口耳传3月,是良心堡镇的油菜花月,君山区第十届油菜花节在这里盛大举行,万亩花海令无数游客流连忘返,印象深刻。与美好同行,与美丽相约,14日,镇禁毒办组织工作人员及志愿者们在景区现场开展禁
漂泊(去海南办厂)九七年九月中旬,按照公司总经理的部署我带领七名员工从上海出发去海口,我们要去那里建一家新厂。作为带队人我要负责大家一路上的饮食安排和安全事项。这次的旅行跟往常不一样,我们每个人除了
坐拥海南最美海岸线,海鲜100元吃到撑,这座小城太浪漫!头条创作挑战赛冬天常常存在着2个惹人注目的远方。一处是寒冬,一处是暖冬。寒冬有雪,暖冬有海。而在春寒料峭之时,最适合抓住冬天的尾巴,奏响春日的乐曲。去海边,去享受松软的沙滩,去触摸
河南新乡积极创建康养旅游示范村资源活了旅游旺了随着旅游市场逐步复苏,河南省新乡市乡村旅游也迎来了开门红,今年12月,全市乡村旅游累计接待游客267。95万人次,持续推动文旅经济又好又快发展。凝心聚力精准指导乡村发展近年来,新乡
芜湖,成长于蚌埠的领土上,逐渐超越蚌埠枢纽的地位,你敢信?芜湖,成长于蚌埠的领土上,逐渐超越蚌埠枢纽的地位,你敢信?早在过去,芜湖是因交通不便,长期被人诟病,难以为继。但是近年来,芜湖的经济和交通业务均得到了快速的发展,逐渐获得了高速推进
可感的生活,无止境的诗意春天的正确打开方式BEAUTIFULLIFE生活之美生活琐碎,万物成诗在高强度快节奏的现代都市生活,我们似乎习惯了凡事讲究效率和目的,好像一切不以结果目的效率为导向的行为,都是在浪费时间浪费生命,于
高坪擦耳5000亩花海荡漾擦耳镇桃花李花梨花油菜花都迎来了盛花期南充新闻网讯(南充日报社全媒体记者卜兵)近日,高坪区擦耳镇四面山村已春意盎然,擦耳镇桃花体量大,除桃花外,李花梨花油菜花也是盛花期,形成多彩的
江苏出境游市场回暖平稳有序随着第二批出境团队游试点国家的公布,出境游复苏步伐加快。近日,江苏省文化和旅游厅在南京举办全省出境组团社业务培训班,对出境游名单填报的依据出境游目的地国家出境游领队备案条件出境社旅
男裙服装中性化与当今男性的困境近期关于男子可否穿着裙装的讨论又多了起来,作为男裙可行性的倡导与实践者,我乐见这样的讨论与争辩,事情越辩越明,一项新生事物的诞生总是艰难而饱受争议的,但风起于青萍之末,我相信独具慧
手机基带芯片简史(上)从2G到3G,高通发家致富的武功秘籍之前看到苹果斥资30亿元买下惠普园区只为尽快用上自研基带芯片的新闻,顿时觉得苹果这太不容易了,为了自研基带芯片,和高通花了45亿美金和解后,又花10亿美金收购了英特尔的基带芯片业务
四川一座尼姑岛,住着2万多的女性修行者,2个禁令让人难接受佛教在中国有着悠久的历史。自传入中国以来,一直在不断发展。今天,佛教是许多国家人民的信仰。在历史时期,佛教的传播也在我国境内留下了许多佛寺佛塔等,是珍贵的佛教文化瑰宝。就像五台山的
洗面奶推荐这些洗面奶能使纤维蛋白再生,让皮肤更加光滑柔软!洗面奶推荐这些洗面奶能使纤维蛋白再生,让皮肤更加光滑柔软!旁氏米萃洗面奶使用感受相比其他洗面奶来说,旁氏主要就是赢在它的价格上,学生党福音了!很大一只,白菜价格,怎么用都不会心疼,
为什么女生腿部内侧皮肤容易色素沉着?俗话说,一白遮百丑,一胖毁所有。自古以来,美白也是所有女性朋友一直追求的永恒话题,相信每个女孩都想拥有一身白皙的皮肤,有助于提升自己的外在形象和气质,增强自信。但是很多女孩腿部内侧
门诊案例儿童抽动症,中医辨证肝肾阴虚,两张中药方治愈!上周视频接诊了一个7岁的小女孩,躁动型抽动症,刚上一年级就被老师叫家长说孩子注意力不集中,上课坐不坐,很多小动作。这个孩子挤眼甩头,喜欢在地上打滚。一个小女孩整天这么躁动,需要从心
春季助长儿童长高黄金期,中医推拿来帮忙!一日之计在于晨一年之计在于春春季是儿童生长发育最快的季节,江西省儿童医院最新出炉了春季助长攻略,带你了解春季儿童助长不迷路。孩子的生长发育不是一蹴而就,需要长期坚持,但春季是助长的
女孩早恋怎样教育才是正确的方法?建议家长看看!多数家长都会有孩子教育方面的问题,特是孩子到青春期,叛逆期的时候,不知道如何跟孩子沟通,又怕担负孩子的成长,特别是女孩早恋怎样教育才是正确的方法的问题。我们每个家长小时候也经历过这
这个方法简简单单调节孩子抑郁情绪之前社会高度关注的胡鑫宇事件,警方调查结果显示,胡鑫宇生前录制的两段音频清晰表达了自杀意愿。而早在胡鑫宇失踪前半个月,他曾向母亲多次通话哭诉,表示不想读书想要回家。然而胡鑫宇的母亲
蓝气球上一道银色斜着能看到春节前,宁波的黄女士购买了一块卡地亚蓝气球系列腕表,说是拿回家5天后,发现表盘内似乎有一道划痕。上个月,黄女士在宁波天一广场的卡地亚腕表店,给自己买了一份新年礼物,就是这块蓝气球系
男人惩罚女人的最好方法是什么图片来自网络,图文无关若是说到惩罚,说明女人碰到了你的底线,你很不服气,很不甘心,为了证明自己不是好惹的,所以打算惩罚对方。惩罚有很多种,有好的和不好的。不好的就是拿自己惩罚别人。
小儿吸痰器的使用方法,你知道吗?小儿吸痰器使用方法1首先要准备一个吸痰器,一个吸痰管,一个滤网,一个灌气管,一个吸痰杯,一个清洗液,以及一些消毒棉等。2把滤网放在吸痰管上,然后将灌气管放在吸痰管上,确保其完全密封
祛斑方法哪种靠谱?专家深度解读,助你早点下斑长沙头条长沙爱尔眼科长斑对于每个爱美人士来说,是一件苦差事。因为斑点别的地方不长,偏偏长到人最显眼的地方脸面上,实属过分,严重违背人爱美的天性。于是一些苦不堪言的上斑族开始盲目寻找