java多线程和并发面试题目(第4题,附答案)
4、ConcurrentLinkedQueue非阻塞无界链表队列
ConcurrentLinkedQueue是一个线程安全的队列,基于链表结构实现,是一个无界队列,理论上来说队列的长度可以无限扩大。
与其他队列相同,ConcurrentLinkedQueue也采用的是先进先出(FIFO)入队规则,对元素进行排序。 (推荐学习:java面试题目)
当我们向队列中添加元素时,新插入的元素会插入到队列的尾部;而当我们获取一个元素时,它会从队列的头部中取出。
因为ConcurrentLinkedQueue是链表结构,所以当入队时,插入的元素依次向后延伸,形成链表;而出队时,则从链表的第一个元素开始获取,依次递增;
值得注意的是,在使用ConcurrentLinkedQueue时,如果涉及到队列是否为空的判断,切记不可使用size()==0的做法,因为在size()方法中,是通过遍历整个链表来实现的,在队列元素很多的时候,size()方法十分消耗性能和时间,只是单纯的判断队列为空使用isEmpty()即可。
public class ConcurrentLinkedQueueTest {<br/> public static int threadCount = 10;<br/> public static ConcurrentLinkedQueue<String> queue = new ConcurrentLinkedQueue<String>();<br/> static class Offer implements Runnable {<br/> public void run() {<br/> //不建议使用 queue.size()==0,影响效率。可以使用!queue.isEmpty()<br/> if (queue.size() == 0) {<br/> String ele = new Random().nextInt(Integer.MAX_VALUE) + "";<br/> queue.offer(ele);<br/> System.out.println("入队元素为" + ele);<br/> }<br/> }<br/> }<br/> static class Poll implements Runnable {<br/> public void run() {<br/> if (!queue.isEmpty()) {<br/> String ele = queue.poll();<br/> System.out.println("出队元素为" + ele);<br/> }<br/> }<br/> }<br/> public static void main(String[] agrs) {<br/> ExecutorService executorService = Executors.newFixedThreadPool(4);<br/> for (int x = 0; x < threadCount; x++) {<br/> executorService.submit(new Offer());<br/> executorService.submit(new Poll());<br/> }<br/> executorService.shutdown();<br/> }<br/>}<br/>
一种输出:
入队元素为313732926<br/>出队元素为313732926<br/>入队元素为812655435<br/>出队元素为812655435<br/>入队元素为1893079357<br/>出队元素为1893079357<br/>入队元素为1137820958<br/>出队元素为1137820958<br/>入队元素为1965962048<br/>出队元素为1965962048<br/>出队元素为685567162<br/>入队元素为685567162<br/>出队元素为1441081163<br/>入队元素为1441081163<br/>出队元素为1627184732<br/>入队元素为1627184732<br/>
ConcurrentLinkedQuere类图
如图ConcurrentLinkedQueue中有两个volatile类型的Node节点分别用来存在列表的首尾节点,其中head节点存放链表第一个item为null的节点,tail则并不是总指向最后一个节点。
Node节点内部则维护一个变量item用来存放节点的值,next用来存放下一个节点,从而链接为一个单向无界列表。
public ConcurrentLinkedQueue(){<br/> head=tail=new Node<E>(null);<br/>}<br/>
如上代码初始化时候会构建一个 item 为 NULL 的空节点作为链表的首尾节点。
Offer 操作offer 操作是在链表末尾添加一个元素,
下面看看实现原理。
public boolean offer(E e) {<br/> //e 为 null 则抛出空指针异常<br/> checkNotNull(e);<br/> //构造 Node 节点构造函数内部调用 unsafe.putObject,后面统一讲<br/> final Node<E> newNode = new Node<E>(e);<br/> //从尾节点插入<br/> for (Node<E> t = tail, p = t; ; ) {<br/> Node<E> q = p.next;<br/> //如果 q=null 说明 p 是尾节点则插入<br/> if (q == null) {<br/> //cas 插入(1)<br/> if (p.casNext(null, newNode)) {<br/> //cas 成功说明新增节点已经被放入链表,然后设置当前尾节点(包含 head,1,3,5.。。个节点为尾节点)<br/> if (p != t)// hop two nodes at a time<br/> casTail(t, newNode); // Failure is OK. return true;<br/> }<br/> // Lost CAS race to another thread; re-read next<br/> } else if (p == q)//(2)<br/> //多线程操作时候,由于 poll 时候会把老的 head 变为自引用,然后 head 的 next 变为新 head,所以这里需要<br/> //重新找新的 head,因为新的 head 后面的节点才是激活的节点<br/> p = (t != (t = tail)) ? t : head;<br/> else<br/> // 寻找尾节点(3)<br/> p = (p != t && t != (t = tail)) ? t : q;<br/> }<br/>}<br/>
从构造函数知道一开始有个item为null的哨兵节点,并且head和tail都是指向这个节点。
以上是java多线程和并发面试题目(第4题,附答案)的详细内容。更多信息请关注PHP中文网其他相关文章!

热AI工具

Undresser.AI Undress
人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover
用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool
免费脱衣服图片

Clothoff.io
AI脱衣机

AI Hentai Generator
免费生成ai无尽的。

热门文章

热工具

记事本++7.3.1
好用且免费的代码编辑器

SublimeText3汉化版
中文版,非常好用

禅工作室 13.0.1
功能强大的PHP集成开发环境

Dreamweaver CS6
视觉化网页开发工具

SublimeText3 Mac版
神级代码编辑软件(SublimeText3)

热门话题

Java 8引入了Stream API,提供了一种强大且表达力丰富的处理数据集合的方式。然而,使用Stream时,一个常见问题是:如何从forEach操作中中断或返回? 传统循环允许提前中断或返回,但Stream的forEach方法并不直接支持这种方式。本文将解释原因,并探讨在Stream处理系统中实现提前终止的替代方法。 延伸阅读: Java Stream API改进 理解Stream forEach forEach方法是一个终端操作,它对Stream中的每个元素执行一个操作。它的设计意图是处

胶囊是一种三维几何图形,由一个圆柱体和两端各一个半球体组成。胶囊的体积可以通过将圆柱体的体积和两端半球体的体积相加来计算。本教程将讨论如何使用不同的方法在Java中计算给定胶囊的体积。 胶囊体积公式 胶囊体积的公式如下: 胶囊体积 = 圆柱体体积 两个半球体体积 其中, r: 半球体的半径。 h: 圆柱体的高度(不包括半球体)。 例子 1 输入 半径 = 5 单位 高度 = 10 单位 输出 体积 = 1570.8 立方单位 解释 使用公式计算体积: 体积 = π × r2 × h (4

Spring Boot简化了可靠,可扩展和生产就绪的Java应用的创建,从而彻底改变了Java开发。 它的“惯例惯例”方法(春季生态系统固有的惯例),最小化手动设置
