dk.clanie.properties.TemporalExpiringProperty Maven / Gradle / Ivy
The newest version!
/**
* Copyright (C) 2010, Claus Nielsen, [email protected]
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License along
* with this program; if not, write to the Free Software Foundation, Inc.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*/
package dk.clanie.properties;
import static dk.clanie.collections.CollectionFactory.newConcurrentSkipListMap;
import java.util.Collections;
import java.util.Map.Entry;
import java.util.NavigableMap;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import org.joda.time.Instant;
/**
* Temporal Property.
*
* Keeps track of the different values assigned to the property over time.
* An implementation of the "Temporal Property" pattern described by
* Martin Fowler (and others).
*
* @author Claus Nielsen
*
* @param the type of the property.
*/
public class TemporalExpiringProperty implements ObservableProperty {
@SuppressWarnings("hiding")
private class ValueListEntry {
Instant expires;
T value;
public ValueListEntry(Instant expires, T value) {
this.expires = expires;
this.value = value;
}
}
public static final Instant EXPIRES_NEVER = new Instant(Long.MAX_VALUE);
NavigableMap> values = newConcurrentSkipListMap();
Set> listeners = Collections.newSetFromMap(new ConcurrentHashMap, Boolean>());
/* (non-Javadoc)
* @see dk.clanie.properties.Property#set(java.lang.Object)
*/
@Override
public void set(T value) {
set(EXPIRES_NEVER, value);
}
/**
* Sets a new value for the property.
*
* The value will expire at the specified point in time, and unless it's superseded by a new value
* it's value will then be null.
*
* @param expires
* @param value
*/
public void set(Instant expires, T value) {
set(new Instant(), expires, value);
}
/**
* Sets the value of the property at the specified time.
*
* For internal and test use only.
* Values are assumed to be set in chronological order.
*
* @param effectiveFrom
* @param value
*/
void set(Instant effectiveFrom, Instant expires, T value) {
T oldValue = get();
values.put(effectiveFrom, new ValueListEntry(expires, value));
notifyListeners(oldValue, value);
}
/* (non-Javadoc)
* @see dk.clanie.properties.Property#get()
*/
@Override
public T get() {
return get(new Instant());
}
/**
* Gets the value effective at the specified instant in time.
*
* Returns null if the property had no value at the specified time.
*
* @param effectivAt
* @return the value of the property at the specified time.
*/
public T get(Instant effectivAt) {
Entry> floorEntry = values.floorEntry(effectivAt);
if (floorEntry == null) return null;
ValueListEntry vle = floorEntry.getValue();
if (!(effectivAt.isBefore(vle.expires))) return null;
return vle.value;
}
/**
* Removes values no longer in effect at the specified point in time.
*
* @param limit
*/
public void purge(Instant limit) {
Instant firstKeeper = values.floorKey(limit);
if (firstKeeper == null) return;
Set purgables = values.headMap(firstKeeper).keySet();
if (purgables == null) return;
for (Instant purgable : purgables) {
values.remove(purgable);
}
}
private void notifyListeners(T oldValue, T newValue) {
for (PropertyChangeListener pcl : listeners) {
pcl.propertyChanged(oldValue, newValue);
}
}
/**
* Adds a new listener to notify when the property's value is changed.
*
* Note, that the listener wil currently NOT be notified when a value expires.
* That may change in a future version.
*
* @see dk.clanie.properties.ObservableProperty#addChangeListener(dk.clanie.properties.PropertyChangeListener)
*/
@Override
public void addChangeListener(PropertyChangeListener listener) {
listeners.add(listener);
}
@Override
public void removeChangeListener(PropertyChangeListener listener) {
listeners.remove(listener);
}
}
© 2015 - 2024 Weber Informatics LLC | Privacy Policy