Do all non-retreival operations on a ConcurrentHashMap (put()
, remove()
etc.) need to be wrapped in a synchronized(this)
block? I und
No, you are losing the benefits of ConcurrentHashMap
by doing that. You may as well be using a HashMap
with synchronized
or synchronizedMap()
to lock the whole table (which is what you do when wrapping operations in synchronized
, since the monitor implied is the entire object instance.)
The purpose of ConcurrentHashMap
is to increase the throughput of your concurrent code by allowing concurrent read/writes on the table without locking the entire table. The table supports this internally by using lock striping (multiple locks instead of one, with each lock assigned to a set of hash buckets - see Java Concurrency in Practice by Goetz et al).
Once you are using ConcurrentHashMap
, all standard map methods (put()
, remove()
, etc.) become atomic by virtue of the lock striping etc. in the implementation. The only tradeoffs are that methods like size()
and isEmpty()
may not necessarily return accurate results, since the only way they could would be for all operations to lock the whole table.
The ConcurrentMap interface interface also adds new atomic compound operations like putIfAbsent()
(put something only if it the key is not already in the map), remove()
accepting both key and value (remove an entry only if its value equals a parameter you pass), etc. These operations used to require locking the whole table because they needed two method calls to accomplish (e.g. putIfAbsent()
would need calls to both containsKey()
and put()
, wrapped inside one synchronized
block, if you were using a standard Map
implementation.) Once again, you gain greater throughput using these methods, by avoiding locking the entire table.