微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

与 ConcurrentSkipListSet 一起正常工作的动态比较器

如何解决与 ConcurrentSkipListSet 一起正常工作的动态比较器

我正在尝试使用

queue = new ConcurrentSkipListSet<Task>(Comparators.comparing(Task::priority))

作为具有唯一元素的并发优先级队列(参见类似的讨论here),但我需要不时更改任务的优先级。

很明显,改变元素的优先级当它们在集合中时就是打开一罐蠕虫;幸运的是,我只需要在从 queue删除它们之后,重新提交它们之前更改它们的优先级。更准确地说,我使用 pollFirst()queue 中弹出一个元素,我可能需要在更新其优先级(优先级较低)后重新提交该元素。

如果这是一个串行实现,当元素在集合之外时改变它们的优先级应该没有问题。

通过并发访问进行此更新的线程安全方式是什么? 是否足以保证

task = queue.pollFirst() 发生在 task.priorityUpdate() 之前,发生在 queue.add(task) 之前?

解决方法

所有并发集合都在元素放置和元素获取之间建立了发生之前的关系。

问题在于,如果您需要在队列中更改优先级,然后将它们取出再放回去,因为这是唯一的方法;然后,并发线程可能会同时放置相同的元素,然后您将丢失修改。在这种情况下,需要进一步同步。

但是如果您要取出元素,更改它们的优先级,然后才评估是否应该将它们放回原处,那么并发集合的发生前保证足以确保正确性,您无需执行任何其他操作。

add()pollFirst()有一个happens-before关系,所以在调用add()的线程中创建的对象对调用pollFirst()的线程是可见的。>

pollFirst()add() 没有。但是,如果您更改优先级,然后从同一线程调用 add(),则不需要进一步的内存限制。
如果您稍后从另一个线程调用 pollFirst(),则 add()pollFirst() 之间的happens-before 关系将保证在调用 add() 之前对对象的更新是可见的。

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


Selenium Web驱动程序和Java。元素在(x,y)点处不可单击。其他元素将获得点击?
Python-如何使用点“。” 访问字典成员?
Java 字符串是不可变的。到底是什么意思?
Java中的“ final”关键字如何工作?(我仍然可以修改对象。)
“loop:”在Java代码中。这是什么,为什么要编译?
java.lang.ClassNotFoundException:sun.jdbc.odbc.JdbcOdbcDriver发生异常。为什么?
这是用Java进行XML解析的最佳库。
Java的PriorityQueue的内置迭代器不会以任何特定顺序遍历数据结构。为什么?
如何在Java中聆听按键时移动图像。
Java“Program to an interface”。这是什么意思?