java - ConcurrentMap 条目同步而不阻塞整个 map

标签 java multithreading

休息服务有许多彼此不相关的客户。 每个客户都可以随时开始投票过程。 一旦客户添加了第一票,我就会在

中创建一个条目
ConcurrentMap<customerId, ConcurrentMap<voteId, voteTimestamp>> votesByCustomer;

每个投票都有 voteTimestamp,所以当它过期时 - 它会被另一个线程从 map 中删除。 (此删除不是简单的 votesByCustomer.remove() 调用,它涉及对 votesByCustomer 映射的多个后续调用,因此必须额外同步)。 因此,上面的 votesByCustomer 映射在所有客户之间共享。 (作为枚举支持的单例中的字段实现)。

此 map 有许多操作,涉及对 votesByCustomer 的多次调用以原子方式完成。所以我使用 syncronized(votesByCustomer) {} 表达式。

因此,当调用这些方法时,它会锁定所有客户。

我的问题是如何以这种方式编写代码,以便当我为一个客户进行操作时它看起来不像其他客户。

我觉得我正在尝试解决一个已经解决的问题,并且有一个完美的 API 和好的方法来解决这个问题,但我无法真正提出适当的 google 请求。你能给一些提示吗?我使用 Java8 和 Spring。

最佳答案

也许你可以使用computeIfAbsent()安全快速地获取与该客户关联的特定 ConcurrentMap(如果不存在则创建一个新的),然后在该客户特定的 上使用 synchronized block ConcurrentMap 用于特定于该客户的所有操作。

我能看到的主要危险是,如果您的投票过期过程从 votesByCustomer 中删除了一个客户条目,那么您最终可能会陷入竞争状态,在这种情况下,您将投票记录到一个不再存在的 ConcurrentHashMap与该客户相关联。因此,您必须确保在其中保留一个空的 ConcurrentMap,即使对于投票已全部过期的客户也是如此。

关于java - ConcurrentMap 条目同步而不阻塞整个 map ,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/45800473/

相关文章:

java - 如何立即从另一个线程使 TestNG 测试失败?

c# - 在主线程中运行代码

java - 如何安装libgdx项目

java - 使用spring任务调度处理多个文件时如何保持一致性?

java - 按 build.xml 依赖项拆分 Java 项目

java - 使用 JENA API - 从 OWL 文件中获取所有类

c++ - 线程与事件同步

multithreading - Powershell:Receive-Job中出现意外结果

java - Selenium Hub 错误地认为 CLIENT_STOPPED_SESSION 已经发生

java - 如何使用 Java 7 解析通配 rune 件规范