 
                        
        
                        
        com.googlecode.blaisemath.util.coordinate.CoordinateManager Maven / Gradle / Ivy
/*
 * CoordinateManager.java
 * Created Oct 5, 2011
 */
package com.googlecode.blaisemath.util.coordinate;
/*
 * #%L
 * BlaiseGraphics
 * --
 * Copyright (C) 2014 - 2017 Elisha Peterson
 * --
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 * 
 *      http://www.apache.org/licenses/LICENSE-2.0
 * 
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 * #L%
 */
import static com.google.common.base.Preconditions.checkNotNull;
import com.google.common.collect.Iterables;
import com.google.common.collect.Lists;
import com.google.common.collect.Maps;
import com.google.common.collect.Sets;
import com.googlecode.blaisemath.annotation.InvokedFromThread;
import java.util.Collection;
import java.util.Collections;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.ConcurrentMap;
import javax.annotation.concurrent.GuardedBy;
import javax.annotation.concurrent.ThreadSafe;
/**
 * 
 * Tracks locations of a collection of objects in a thread-safe manner.
 * Maintains a cache of prior locations, so that if some of the objects are removed,
 * this class "remembers" their prior locations. Listeners may register to be notified
 * when any of the coordinates within the manager change, or when any objects are
 * added to or removed from the manager.
 * 
 * 
 * The object is thread safe, so the points in the manager can be read from or written to
 * by multiple threads. Thread safety involves managing access to three interdependent
 * state variables, representing the cached locations, the objects that are "active" and
 * the objects that are "inactive". It is fine to iterate over these sets from any thread,
 * although they may change during iteration.
 * 
 * 
 * Care should be taken with event handlers to ensure thread safety. Listeners
 * registering for {@link CoordinateChangeEvent}s are notified of the change from
 * the thread that makes the change. Collections passed with the event will be
 * either immutable copies, or references passed to this object as parameters to
 * a mutator method.
 * 
 *
 * @param  type of source object
 * @param  type of point
 *
 * @author Elisha Peterson
 */
@ThreadSafe
public final class CoordinateManager {
    
    /** Max size of the cache */
    private final int maxCacheSize;
    
    /** Map with current objects and locations (stores the data) */
    @GuardedBy("this")
    private final ConcurrentMap map = Maps.newConcurrentMap();
    /** Active objects. This value may be set. */
    @GuardedBy("this")
    private Set active = Sets.newConcurrentHashSet();
    /** Cached objects */
    @GuardedBy("this")
    private final Set inactive = Sets.newConcurrentHashSet();
    /** Listeners that will receive updates. */
    private final List listeners = Lists.newCopyOnWriteArrayList();
    
    private CoordinateManager(int maxCacheSize) {
        this.maxCacheSize = maxCacheSize;
    }
    
    //
    
    /**
     * Create and return new instance of coordinate manager.
     * @param  type of source object
     * @param  type of point
     * @param maxCacheSize maximum # of active and inactive points to include
     * @return newly created coordinate manager.
     */
    public static  CoordinateManager create(int maxCacheSize) {
        return new CoordinateManager(maxCacheSize);
    }
    
    //  
    public int getMaxCacheSize() {
        return maxCacheSize;
    }
    /**
     * Return objects currently tracked by the manager.
     * @return objects
     */
    public Set getActive() {
        return Collections.unmodifiableSet(active);
    }
    /**
     * Returns cached objects.
     * @return cached objects
     */
    public Set getInactive() {
        return Collections.unmodifiableSet(inactive);
    }
    /**
     * Tests to see if all provided items are contained in either current
     * locations or cached locations.
     * @param obj objects to test
     * @return true if all are tracked, false otherwise
     */
    public boolean locatesAll(Collection extends S> obj) {
        return map.keySet().containsAll(obj);
    }
    /**
     * Returns copy of map with active locations. This method blocks on the entire
     * cache, since it uses both state variables.
     * @return object locations
     */
    public synchronized Map getActiveLocationCopy() {
        Map res = Maps.newHashMap();
        for (S s : active) {
            res.put(s, map.get(s));
        }
        return res;
    }
    
    /**
     * Retrieve location of a single point, whether active or inactive.
     * @param obj object to retrieve
     * @return location
     */
    public C getLocation(S obj) {
        return map.get(obj);
    }
    
    /**
     * Retrieve location of given set of objects, whether active or inactive.
     * @param  type of object in provided set
     * @param obj objects to retrieve
     * @return map of locations
     */
    public  Map getLocationCopy(Set obj) {
        synchronized(map) {
            Map res = Maps.newHashMap();
            for (S s : obj) {
                res.put(s, map.get(s));
            }
            return res;
        }
    }
    /**
     * Returns copy of map with inactive locations. This method blocks on the entire
     * cache, since it uses both state variables.
     * @return object locations
     */
    public synchronized Map getInactiveLocationCopy() {
        Map res = Maps.newHashMap();
        for (S s : inactive) {
            res.put(s, map.get(s));
        }
        return res;
    }
    //
    // MUTATORS
    //
    /**
     * Adds a single additional location to the manager. Use {@link #putAll(java.util.Map)}
     * wherever possible as it will be more efficient.
     * @param s source object
     * @param c coordinate
     */
    public void put(S s, C c) {
        putAll(Collections.singletonMap(s, c));
    }
    /**
     * Adds additional locations to the manager. Blocks while the map is being
     * updated, since it may change the active and cached object sets.
     * Propagates the updated coordinates to interested listeners (on the invoking thread).
     * @param coords new coordinates
     */
    public void putAll(Map coords) {
        Map coordCopy = Maps.newHashMap(coords);
        synchronized (this) {
            map.putAll(coordCopy);
            active.addAll(coordCopy.keySet());
            inactive.removeAll(coordCopy.keySet());
        }
        fireCoordinatesChanged(CoordinateChangeEvent.createAddEvent(this, coordCopy));
    }
    /**
     * Replaces the current set of objects with specified objects, and caches the rest.
     * Propagates the updated coordinates to interested listeners (on the invoking thread).
     * @param coords new coordinates
     */
    public void setCoordinateMap(Map coords) {
        Map coordCopy = Maps.newHashMap(coords);
        Set toCache;
        synchronized(this) {
            toCache = Sets.difference(map.keySet(), coordCopy.keySet()).immutableCopy();
            map.putAll(coordCopy);
            active = Sets.newConcurrentHashSet(coordCopy.keySet());
            inactive.removeAll(coordCopy.keySet());
            inactive.addAll(toCache);
            checkCache();
        }
        fireCoordinatesChanged(CoordinateChangeEvent.createAddRemoveEvent(this, coordCopy, toCache));
    }
    /**
     * Removes objects from the manager without caching their locations.
     * Propagates the updated coordinates to interested listeners (on the invoking thread).
     * @param obj objects to remove
     */
    public void forget(Set extends S> obj) {
        Set removed = new HashSet();
        synchronized (map) {
            for (S k : obj) {
                if (map.remove(k) != null) {
                    removed.add(k);
                }
            }
        }
        fireCoordinatesChanged(CoordinateChangeEvent.createRemoveEvent(this, removed));
    }
    /**
     * Makes specified objects inactive, possibly removing them from memory.
     * Propagates the updated coordinates to interested listeners (on the invoking thread).
     * @param  type of object in provided set
     * @param obj objects to removeObjects
     */
    public  void deactivate(Set obj) {
        Set removed;
        synchronized (this) {
            removed = Sets.intersection(obj, active).immutableCopy();
            active.removeAll(removed);
            inactive.addAll(removed);
            checkCache();
        }
        fireCoordinatesChanged(CoordinateChangeEvent.createRemoveEvent(this, removed));
    }
    /**
     * Call to restore locations from the cache.
     * @param  type of object in provided set
     * @param obj objects to restore
     * @return true if cache was changed
     */
    public  boolean reactivate(Set obj) {
        Map restoreMap = Maps.newHashMap();
        synchronized (this) {
            Set restored = Sets.intersection(obj, inactive);
            for (T t : restored) {
                restoreMap.put(t, map.get(t));
            }
            active.addAll(restored);
            inactive.removeAll(restored);
        }
        fireCoordinatesChanged(CoordinateChangeEvent.createAddEvent(this, restoreMap));
        return !restoreMap.isEmpty();
    }
    /** 
     * Call to ensure appropriate size of cache. Should always be called within
     * a synchronization block.
     */
    private void checkCache() {
        int n = inactive.size() - maxCacheSize;
        if (n > 0) {
            Set remove = Sets.newHashSet(Iterables.limit(inactive, n));
            inactive.removeAll(remove);
            map.keySet().removeAll(remove);
        }
    }
    //
    //
    // EVENT HANDLING
    //
    /**
     * Fire update, from the thread that invoked the change.
     * The collections in the event are either provided as arguments to
     * {@code this}, or are immutable lists, and therefore may be used freely
     * from any thread.
     * 
     * @param evt the event to fire
     */
    @InvokedFromThread("unknown")
    protected final void fireCoordinatesChanged(CoordinateChangeEvent evt) {
        Map added = evt.getAdded();
        Set removed = evt.getRemoved();
        if ((added == null || added.isEmpty()) && (removed == null || removed.isEmpty())) {
            return;
        }
        for (CoordinateListener cl : listeners) {
            cl.coordinatesChanged(evt);
        }
    }
    public final void addCoordinateListener(CoordinateListener cl) {
        checkNotNull(cl);
        listeners.add(cl);
    }
    public final void removeCoordinateListener(CoordinateListener cl) {
        checkNotNull(cl);
        listeners.remove(cl);
    }
    // 
}