首页
|
新闻
|
新品
|
文库
|
方案
|
视频
|
下载
|
商城
|
开发板
|
数据中心
|
座谈新版
|
培训
|
工具
|
博客
|
论坛
|
百科
|
GEC
|
活动
|
主题月
|
电子展
注册
登录
论坛
博客
搜索
帮助
导航
默认风格
uchome
discuz6
GreenM
»
MCU 单片机技术
»
PowerPC
» Java 容器源码分析之ArrayBlockingQueue和LinkedBlockingQueue(4)
返回列表
回复
发帖
发新话题
发布投票
发布悬赏
发布辩论
发布活动
发布视频
发布商品
Java 容器源码分析之ArrayBlockingQueue和LinkedBlockingQueue(4)
发短消息
加为好友
look_w
当前离线
UID
1066743
帖子
8283
精华
0
积分
4142
阅读权限
90
在线时间
233 小时
注册时间
2017-6-23
最后登录
2019-5-18
论坛元老
UID
1066743
1
#
打印
字体大小:
t
T
look_w
发表于 2019-1-11 18:35
|
只看该作者
Java 容器源码分析之ArrayBlockingQueue和LinkedBlockingQueue(4)
数据的删除LinkedBlockingQueue有不同的几个数据删除方法,poll、take、remove方法。
poll方法:
[url=]
[/url]
public
E poll() {
final
AtomicInteger count =
this
.count;
if
(count.get() == 0)
//
如果元素个数为0
return
null
;
//
返回null
E x =
null
;
int
c = -1
;
final
ReentrantLock takeLock =
this
.takeLock; takeLock.lock();
//
拿锁加锁,保证调用poll方法的时候只有1个线程
try
{
if
(count.get() > 0) {
//
判断队列里是否还有数据
x = dequeue();
//
删除头结点
c = count.getAndDecrement();
//
元素个数-1
if
(c > 1)
//
如果队列里还有元素
notEmpty.signal();
//
在拿锁的条件对象notEmpty上唤醒正在等待的线程,表示队列里还有数据,可以再次消费
} }
finally
{ takeLock.unlock();
//
释放拿锁,让其他线程可以调用poll方法
}
if
(c == capacity)
//
由于存在放锁和拿锁,这里可能放锁一直在添加数据,count会变化。这里的if条件表示如果队列中还可以再插入数据
signalNotFull();
//
在放锁的条件对象notFull上唤醒正在等待的1个线程,表示队列里还能再次添加数据
return
x;}
[url=]
[/url]
take方法:[url=]
[/url]
public
E take()
throws
InterruptedException { E x;
int
c = -1
;
final
AtomicInteger count =
this
.count;
final
ReentrantLock takeLock =
this
.takeLock; takeLock.lockInterruptibly();
//
拿锁加锁,保证调用take方法的时候只有1个线程
try
{
while
(count.get() == 0) {
//
如果队列里已经没有元素了
notEmpty.await();
//
阻塞并挂起当前线程
} x
= dequeue();
//
删除头结点
c = count.getAndDecrement();
//
元素个数-1
if
(c > 1)
//
如果队列里还有元素
notEmpty.signal();
//
在拿锁的条件对象notEmpty上唤醒正在等待的线程,表示队列里还有数据,可以再次消费
}
finally
{ takeLock.unlock();
//
释放拿锁,让其他线程可以调用take方法
}
if
(c == capacity)
//
由于存在放锁和拿锁,这里可能放锁一直在添加数据,count会变化。这里的if条件表示如果队列中还可以再插入数据
signalNotFull();
//
在放锁的条件对象notFull上唤醒正在等待的1个线程,表示队列里还能再次添加数据
return
x;}
[url=]
[/url]
remove方法:
[url=]
[/url]
public
boolean
remove(Object o) {
if
(o ==
null
)
return
false
; fullyLock();
//
remove操作要移动的位置不固定,2个锁都需要加锁
try
{
for
(Node<E> trail = head, p = trail.next;
//
从链表头结点开始遍历
p !=
null
; trail
= p, p =
p.next) {
if
(o.equals(p.item)) {
//
判断是否找到对象
unlink(p, trail);
//
修改节点的链接信息,同时调用notFull的signal方法
return
true
; } }
return
false
; }
finally
{ fullyUnlock();
//
2个锁解锁
}}
[url=]
[/url]
LinkedBlockingQueue的take方法对于没数据的情况下会阻塞,poll方法删除链表头结点,remove方法删除指定的对象。
需要注意的是remove方法由于要删除的数据的位置不确定,需要2个锁同时加锁。
收藏
分享
评分
回复
引用
订阅
TOP
返回列表
电商论坛
Pine A64
资料下载
方案分享
FAQ
行业应用
消费电子
便携式设备
医疗电子
汽车电子
工业控制
热门技术
智能可穿戴
3D打印
智能家居
综合设计
示波器技术
存储器
电子制造
计算机和外设
软件开发
分立器件
传感器技术
无源元件
资料共享
PCB综合技术
综合技术交流
EDA
MCU 单片机技术
ST MCU
Freescale MCU
NXP MCU
新唐 MCU
MIPS
X86
ARM
PowerPC
DSP技术
嵌入式技术
FPGA/CPLD可编程逻辑
模拟电路
数字电路
富士通半导体FRAM 铁电存储器“免费样片”使用心得
电源与功率管理
LED技术
测试测量
通信技术
3G
无线技术
微波在线
综合交流区
职场驿站
活动专区
在线座谈交流区
紧缺人才培训课程交流区
意见和建议