Search code examples
javamultithreadingsynchronizationconcurrenthashmap

Thread safety when iterating through concurrent collections


I'm writing some client-server-application where I have to deal with multiple threads. I've got some servers, that send alive-packets every few seconds. Those servers are maintained in a ConcurrentHashMap, that contains their EndPoints paired with the time the last alive-package arrived of the respective server.

Now I've got a thread, that has to "sort out" all the servers that haven't sent alive-packets for a specific amount of time.

I guess I can't just do it like that, can I?

for( IPEndPoint server : this.fileservers.keySet() )
{
    Long time = this.fileservers.get( server );

    //If server's time is updated here, I got a problem

    if( time > fileserverTimeout )
        this.fileservers.remove( server );
}

Is there a way I can get around that without aquiring a lock for the whole loop (that I then have to respect in the other threads as well)?


Solution

  • There is probably no problem here, depending on what exactly you store in the map. Your code looks a little weird to me, since you seem to save "the duration for which the server hasn't been active".

    My first idea for recording that data was to store "the latest timestamp at which the server has been active". Then your code would look like this:

    package so3950354;
    
    import java.util.Iterator;
    import java.util.Map;
    import java.util.concurrent.ConcurrentHashMap;
    import java.util.concurrent.ConcurrentMap;
    
    public class ServerManager {
    
      private final ConcurrentMap<Server, Long> lastActive = new ConcurrentHashMap<Server, Long>();
    
      /** May be overridden by a special method for testing. */
      protected long now() {
        return System.currentTimeMillis();
      }
    
      public void markActive(Server server) {
        lastActive.put(server, Long.valueOf(now()));
      }
    
      public void removeInactive(long timeoutMillis) {
        final long now = now();
    
        Iterator<Map.Entry<Server, Long>> it = lastActive.entrySet().iterator();
        while (it.hasNext()) {
          final Map.Entry<Server, Long> entry = it.next();
          final long backThen = entry.getValue().longValue();
          /*
           * Even if some other code updates the timestamp of this server now,
           * the server had timed out at some point in time, so it may be
           * removed. It's bad luck, but impossible to avoid.
           */
          if (now - backThen >= timeoutMillis) {
            it.remove();
          }
        }
      }
    
      static class Server {
    
      }
    }
    

    If you really want to avoid that no code ever calls markActive during a call to removeInactive, there is no way around explicit locking. What you probably want is:

    • concurrent calls to markActive are allowed.
    • during markActive no calls to removeInactive are allowed.
    • during removeInactive no calls to markActive are allowed.

    This looks like a typical scenario for a ReadWriteLock, where markActive is the "reading" operation and removeInactive is the "writing" Operation.