eclipse的servlet默认不执行index_Java中常用数据结构执行过程及原理——动图+源码...

最近在整理数据结构方面的知识, 系统化看了下Java中常用数据结构, 突发奇想用动画来绘制数据流转过程.

主要基于jdk8, 可能会有些特性与jdk7之前不相同, 例如LinkedList LinkedHashMap中的双向列表不再是回环的.

HashMap中的单链表是尾插, 而不是头插入等等, 后文不再赘叙这些差异, 本文目录结构如下:

ea7545ffc6943058cf4409c5f494ba1b.png

LinkedList

经典的双链表结构, 适用于乱序插入, 删除. 指定序列操作则性能不如ArrayList, 这也是其数据结构决定的.

add(E) / addLast(E)

368c95b940486674919cd16f698b00df.gif

add(index, E)

这边有个小的优化, 他会先判断index是靠近队头还是队尾, 来确定从哪个方向遍历链入.

52bca0ef008fa38b3345110832e4824b.png
2d3c57008f389d9ebfaa764273fc4d0f.gif

靠队尾

8bb319f15c1f855484949b37d5fb51a4.gif

get(index)

也是会先判断index, 不过性能依然不好, 这也是为什么不推荐用for(int i = 0; i < lengh; i++)的方式遍历linkedlist, 而是使用iterator的方式遍历.

43e8991d547e50e76f315598ffb82e7f.gif
6ae047687b228c2c7ddf3bb12b10ab5d.gif

remove(E)

78929aaf1cbe115e79a1ed6592a921aa.gif

ArrayList

底层就是一个数组, 因此按序查找快, 乱序插入, 删除因为涉及到后面元素移位所以性能慢.

add(index, E)

5582b3bcb3beec826b6f1f63d30527be.gif

扩容

一般默认容量是10, 扩容后, 会length*1.5.

ab00f403c5845d839e2809aa10f90b7c.gif

remove(E)

循环遍历数组, 判断E是否equals当前元素, 删除性能不如LinkedList.

98845740f0c11db56287240c1761dc55.gif

Stack

经典的数据结构, 底层也是数组, 继承自Vector, 先进后出FILO, 默认new Stack()容量为10, 超出自动扩容.

push(E)

555c564a034236176ee33adab67110e3.gif

pop()

4d79c3ed4302fdff902e81570e9a3b67.gif

后缀表达式

Stack的一个典型应用就是计算表达式如 9 + (3 - 1) * 3 + 10 / 2, 计算机将中缀表达式转为后缀表达式, 再对后缀表达式进行计算.

中缀转后缀

数字直接输出

栈为空时,遇到运算符,直接入栈

遇到左括号, 将其入栈

遇到右括号, 执行出栈操作,并将出栈的元素输出,直到弹出栈的是左括号,左括号不输出。

遇到运算符(加减乘除):弹出所有优先级大于或者等于该运算符的栈顶元素,然后将该运算符入栈

最终将栈中的元素依次出栈,输出。

e071d4d0956cd1ff34da19dcc7f938a6.gif

计算后缀表达

遇到数字时,将数字压入堆栈

遇到运算符时,弹出栈顶的两个数,用运算符对它们做相应的计算, 并将结果入栈

重复上述过程直到表达式最右端

运算得出的值即为表达式的结果

a5b8f981e401fc2f454e33b2fa362c6a.gif

队列

与Stack的区别在于, Stack的删除与添加都在队尾进行, 而Queue删除在队头, 添加在队尾.

ArrayBlockingQueue

生产消费者中常用的阻塞有界队列, FIFO.

put(E)

150d1774b301a4b289d49c764ee6f1d6.gif

put(E) 队列满了

0a7883e61b97ce515efca8fccf5b7d87.png
e34ece16501d5c3d3dcef5f5babc21ec.gif

take()

当元素被取出后, 并没有对数组后面的元素位移, 而是更新takeIndex来指向下一个元素.

takeIndex是一个环形的增长, 当移动到队列尾部时, 会指向0, 再次循环.

ccf02a6ad5e4ed502deba71e977f88dc.png
a78ca70c5d03758e59b83f31ac5ca1b2.gif

HashMap

最常用的哈希表, 面试的童鞋必备知识了, 内部通过数组 + 单链表的方式实现. jdk8中引入了红黑树对长度 > 8的链表进行优化, 我们另外篇幅再讲.

put(K, V)

626859f216d9d39bcd87fd678fd5b2ff.gif

put(K, V) 相同hash值

e31abd943a3c25f666f26b122f1129f0.gif

resize 动态扩容

当map中元素超出设定的阈值后, 会进行resize (length * 2)操作, 扩容过程中对元素一通操作, 并放置到新的位置.

具体操作如下:

在jdk7中对所有元素直接rehash, 并放到新的位置.

在jdk8中判断元素原hash值新增的bit位是0还是1, 0则索引不变, 1则索引变成"原索引 + oldTable.length".

22cd6c09f5afb0a5b7c2ba82b87e4c59.png
c1521b112db933e17a10fb916ae03021.gif

LinkedHashMap

继承自HashMap, 底层额外维护了一个双向链表来维持数据有序. 可以通过设置accessOrder来实现FIFO(插入有序)或者LRU(访问有序)缓存.

put(K, V)

04e803961f1cea18847d8d2d53573aab.gif

get(K)

accessOrder为false的时候, 直接返回元素就行了, 不需要调整位置.

accessOrder为true的时候, 需要将最近访问的元素, 放置到队尾.

9e059c2b90be53b49035881f6acb0af5.gif

removeEldestEntry 删除最老的元素

643516def4acd5b0d2bdd1ec14107436.gif

最新整理的Java技术干货文档资料:【Java核心知识点整理】涵盖29个Java核心技术详解,JVM,Redis,Nginx,Spring Boot,Spring Cloud,Kafka,并发编程,Tomcat,MyBatis,BAT面试题,Java技术精讲视频等。转发+关注,私信回复“干货”即可获得免费领取方式。


链接:https://www.jianshu.com/p/2cfaa2da1d16

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值