休息服务有许多彼此不相关的客户。 每个客户都可以随时开始投票过程。 一旦客户添加了第一票,我就会在
中创建一个条目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/