Add putIfAbsent on Cache abstraction

This commit adds a putIfAbsent method to the Cache interface. This
method offers an atomic put if the key is not already associated in
the cache.

Issue: SPR-11400
This commit is contained in:
Stephane Nicoll 2014-02-12 14:41:34 +01:00
parent 8ed490c4d7
commit 3e74d3b2fb
12 changed files with 332 additions and 54 deletions

View File

@ -29,6 +29,7 @@ import org.springframework.util.Assert;
* *
* @author Costin Leau * @author Costin Leau
* @author Juergen Hoeller * @author Juergen Hoeller
* @author Stephane Nicoll
* @since 3.1 * @since 3.1
*/ */
public class EhCacheCache implements Cache { public class EhCacheCache implements Cache {
@ -62,7 +63,7 @@ public class EhCacheCache implements Cache {
@Override @Override
public ValueWrapper get(Object key) { public ValueWrapper get(Object key) {
Element element = this.cache.get(key); Element element = this.cache.get(key);
return (element != null ? new SimpleValueWrapper(element.getObjectValue()) : null); return toWrapper(element);
} }
@Override @Override
@ -81,6 +82,12 @@ public class EhCacheCache implements Cache {
this.cache.put(new Element(key, value)); this.cache.put(new Element(key, value));
} }
@Override
public ValueWrapper putIfAbsent(Object key, Object value) {
Element existingElement = this.cache.putIfAbsent(new Element(key, value));
return toWrapper(existingElement);
}
@Override @Override
public void evict(Object key) { public void evict(Object key) {
this.cache.remove(key); this.cache.remove(key);
@ -91,4 +98,8 @@ public class EhCacheCache implements Cache {
this.cache.removeAll(); this.cache.removeAll();
} }
private ValueWrapper toWrapper(Element element) {
return (element != null ? new SimpleValueWrapper(element.getObjectValue()) : null);
}
} }

View File

@ -17,6 +17,8 @@
package org.springframework.cache.guava; package org.springframework.cache.guava;
import java.io.Serializable; import java.io.Serializable;
import java.util.concurrent.Callable;
import java.util.concurrent.ExecutionException;
import org.springframework.cache.Cache; import org.springframework.cache.Cache;
import org.springframework.cache.support.SimpleValueWrapper; import org.springframework.cache.support.SimpleValueWrapper;
@ -29,6 +31,7 @@ import org.springframework.util.Assert;
* <p>Requires Google Guava 12.0 or higher. * <p>Requires Google Guava 12.0 or higher.
* *
* @author Juergen Hoeller * @author Juergen Hoeller
* @author Stephane Nicoll
* @since 4.0 * @since 4.0
*/ */
public class GuavaCache implements Cache { public class GuavaCache implements Cache {
@ -83,7 +86,7 @@ public class GuavaCache implements Cache {
@Override @Override
public ValueWrapper get(Object key) { public ValueWrapper get(Object key) {
Object value = this.cache.getIfPresent(key); Object value = this.cache.getIfPresent(key);
return (value != null ? new SimpleValueWrapper(fromStoreValue(value)) : null); return toWrapper(value);
} }
@Override @Override
@ -101,6 +104,17 @@ public class GuavaCache implements Cache {
this.cache.put(key, toStoreValue(value)); this.cache.put(key, toStoreValue(value));
} }
@Override
public ValueWrapper putIfAbsent(Object key, final Object value) {
try {
PutIfAbsentCallable callable = new PutIfAbsentCallable(value);
Object result = this.cache.get(key, callable);
return (callable.called ? null : toWrapper(result));
} catch (ExecutionException e) {
throw new IllegalArgumentException(e);
}
}
@Override @Override
public void evict(Object key) { public void evict(Object key) {
this.cache.invalidate(key); this.cache.invalidate(key);
@ -138,9 +152,29 @@ public class GuavaCache implements Cache {
return userValue; return userValue;
} }
private ValueWrapper toWrapper(Object value) {
return (value != null ? new SimpleValueWrapper(fromStoreValue(value)) : null);
}
@SuppressWarnings("serial") @SuppressWarnings("serial")
private static class NullHolder implements Serializable { private static class NullHolder implements Serializable {
} }
private class PutIfAbsentCallable implements Callable<Object> {
private boolean called;
private final Object value;
private PutIfAbsentCallable(Object value) {
this.value = value;
}
@Override
public Object call() throws Exception {
called = true;
return toStoreValue(value);
}
}
} }

View File

@ -29,6 +29,7 @@ import org.springframework.util.Assert;
* <p>Note: This class has been updated for JCache 1.0, as of Spring 4.0. * <p>Note: This class has been updated for JCache 1.0, as of Spring 4.0.
* *
* @author Juergen Hoeller * @author Juergen Hoeller
* @author Stephane Nicoll
* @since 3.2 * @since 3.2
*/ */
public class JCacheCache implements Cache { public class JCacheCache implements Cache {
@ -95,6 +96,12 @@ public class JCacheCache implements Cache {
this.cache.put(key, toStoreValue(value)); this.cache.put(key, toStoreValue(value));
} }
@Override
public ValueWrapper putIfAbsent(Object key, Object value) {
boolean set = this.cache.putIfAbsent(key, toStoreValue(value));
return (set ? null : get(key));
}
@Override @Override
public void evict(Object key) { public void evict(Object key) {
this.cache.remove(key); this.cache.remove(key);

View File

@ -1,5 +1,5 @@
/* /*
* Copyright 2002-2013 the original author or authors. * Copyright 2002-2014 the original author or authors.
* *
* Licensed under the Apache License, Version 2.0 (the "License"); * Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License. * you may not use this file except in compliance with the License.
@ -28,7 +28,11 @@ import org.springframework.util.Assert;
* successful transaction. If no transaction is active, {@link #put} and {@link #evict} * successful transaction. If no transaction is active, {@link #put} and {@link #evict}
* operations will be performed immediately, as usual. * operations will be performed immediately, as usual.
* *
* <p>Use of more aggressive operations such as {@link #putIfAbsent} cannot be deferred
* to the after-commit phase of a running transaction. Use these with care.
*
* @author Juergen Hoeller * @author Juergen Hoeller
* @author Stephane Nicoll
* @since 3.2 * @since 3.2
* @see TransactionAwareCacheManagerProxy * @see TransactionAwareCacheManagerProxy
*/ */
@ -82,6 +86,11 @@ public class TransactionAwareCacheDecorator implements Cache {
} }
} }
@Override
public ValueWrapper putIfAbsent(final Object key, final Object value) {
return this.targetCache.putIfAbsent(key, value);
}
@Override @Override
public void evict(final Object key) { public void evict(final Object key) {
if (TransactionSynchronizationManager.isSynchronizationActive()) { if (TransactionSynchronizationManager.isSynchronizationActive()) {

View File

@ -0,0 +1,105 @@
/*
* Copyright 2002-2014 the original author or authors.
*
* 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.
*/
package org.springframework.cache;
import static org.junit.Assert.*;
import org.junit.Test;
/**
* @author Stephane Nicoll
*/
public abstract class AbstractCacheTests<T extends Cache> {
protected final static String CACHE_NAME = "testCache";
protected abstract T getCache();
protected abstract Object getNativeCache();
@Test
public void testCacheName() throws Exception {
assertEquals(CACHE_NAME, getCache().getName());
}
@Test
public void testNativeCache() throws Exception {
assertSame(getNativeCache(), getCache().getNativeCache());
}
@Test
public void testCachePut() throws Exception {
T cache = getCache();
Object key = "enescu";
Object value = "george";
assertNull(cache.get(key));
assertNull(cache.get(key, String.class));
assertNull(cache.get(key, Object.class));
cache.put(key, value);
assertEquals(value, cache.get(key).get());
assertEquals(value, cache.get(key, String.class));
assertEquals(value, cache.get(key, Object.class));
assertEquals(value, cache.get(key, null));
cache.put(key, null);
assertNotNull(cache.get(key));
assertNull(cache.get(key).get());
assertNull(cache.get(key, String.class));
assertNull(cache.get(key, Object.class));
}
@Test
public void testCachePutIfAbsent() throws Exception {
T cache = getCache();
Object key = new Object();
Object value = "initialValue";
assertNull(cache.get(key));
assertNull(cache.putIfAbsent(key, value));
assertEquals(value, cache.get(key).get());
assertEquals("initialValue", cache.putIfAbsent(key, "anotherValue").get());
assertEquals(value, cache.get(key).get()); // not changed
}
@Test
public void testCacheRemove() throws Exception {
T cache = getCache();
Object key = "enescu";
Object value = "george";
assertNull(cache.get(key));
cache.put(key, value);
}
@Test
public void testCacheClear() throws Exception {
T cache = getCache();
assertNull(cache.get("enescu"));
cache.put("enescu", "george");
assertNull(cache.get("vlaicu"));
cache.put("vlaicu", "aurel");
cache.clear();
assertNull(cache.get("vlaicu"));
assertNull(cache.get("enescu"));
}
}

View File

@ -16,56 +16,55 @@
package org.springframework.cache.ehcache; package org.springframework.cache.ehcache;
import static org.junit.Assert.*;
import net.sf.ehcache.CacheManager; import net.sf.ehcache.CacheManager;
import net.sf.ehcache.Ehcache; import net.sf.ehcache.Ehcache;
import net.sf.ehcache.Element; import net.sf.ehcache.Element;
import net.sf.ehcache.config.CacheConfiguration; import net.sf.ehcache.config.CacheConfiguration;
import net.sf.ehcache.config.Configuration;
import org.junit.After;
import org.junit.Before; import org.junit.Before;
import org.junit.Test; import org.junit.Test;
import org.springframework.cache.AbstractCacheTests;
import org.springframework.cache.Cache;
import org.springframework.tests.Assume; import org.springframework.tests.Assume;
import org.springframework.tests.TestGroup; import org.springframework.tests.TestGroup;
import static org.junit.Assert.*;
/** /**
* @author Costin Leau * @author Costin Leau
* @author Stephane Nicoll
* @author Juergen Hoeller * @author Juergen Hoeller
*/ */
public class EhCacheCacheTests { public class EhCacheCacheTests extends AbstractCacheTests<EhCacheCache> {
protected final static String CACHE_NAME = "testCache";
protected Ehcache nativeCache;
protected Cache cache;
private CacheManager cacheManager;
private Ehcache nativeCache;
private EhCacheCache cache;
@Before @Before
public void setUp() throws Exception { public void setUp() {
if (CacheManager.getInstance().cacheExists(CACHE_NAME)) { cacheManager = new CacheManager(new Configuration().name("EhCacheCacheTests")
nativeCache = CacheManager.getInstance().getEhcache(CACHE_NAME); .defaultCache(new CacheConfiguration("default", 100)));
} nativeCache = new net.sf.ehcache.Cache(new CacheConfiguration(CACHE_NAME, 100));
else { cacheManager.addCache(nativeCache);
nativeCache = new net.sf.ehcache.Cache(new CacheConfiguration(CACHE_NAME, 100));
CacheManager.getInstance().addCache(nativeCache);
}
cache = new EhCacheCache(nativeCache); cache = new EhCacheCache(nativeCache);
cache.clear();
} }
@After
@Test public void tearDown() {
public void testCacheName() throws Exception { cacheManager.shutdown();
assertEquals(CACHE_NAME, cache.getName());
} }
@Test @Override
public void testNativeCache() throws Exception { protected EhCacheCache getCache() {
assertSame(nativeCache, cache.getNativeCache()); return cache;
} }
@Override
protected Ehcache getNativeCache() {
return nativeCache;
}
@Test @Test
public void testCachePut() throws Exception { public void testCachePut() throws Exception {
Object key = "enescu"; Object key = "enescu";
@ -88,26 +87,6 @@ public class EhCacheCacheTests {
assertNull(cache.get(key, Object.class)); assertNull(cache.get(key, Object.class));
} }
@Test
public void testCacheRemove() throws Exception {
Object key = "enescu";
Object value = "george";
assertNull(cache.get(key));
cache.put(key, value);
}
@Test
public void testCacheClear() throws Exception {
assertNull(cache.get("enescu"));
cache.put("enescu", "george");
assertNull(cache.get("vlaicu"));
cache.put("vlaicu", "aurel");
cache.clear();
assertNull(cache.get("vlaicu"));
assertNull(cache.get("enescu"));
}
@Test @Test
public void testExpiredElements() throws Exception { public void testExpiredElements() throws Exception {
Assume.group(TestGroup.LONG_RUNNING); Assume.group(TestGroup.LONG_RUNNING);
@ -123,5 +102,4 @@ public class EhCacheCacheTests {
Thread.sleep(5 * 1000); Thread.sleep(5 * 1000);
assertNull(cache.get(key)); assertNull(cache.get(key));
} }
} }

View File

@ -0,0 +1,67 @@
/*
* Copyright 2002-2014 the original author or authors.
*
* 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.
*/
package org.springframework.cache.guava;
import static org.junit.Assert.*;
import com.google.common.cache.CacheBuilder;
import org.junit.Before;
import org.junit.Test;
import org.springframework.cache.AbstractCacheTests;
import org.springframework.cache.Cache;
/**
* @author Stephane Nicoll
*/
public class GuavaCacheTests extends AbstractCacheTests<GuavaCache> {
private com.google.common.cache.Cache<Object, Object> nativeCache;
private GuavaCache cache;
@Before
public void setUp() {
nativeCache = CacheBuilder.newBuilder().build();
cache = new GuavaCache(CACHE_NAME, nativeCache);
}
@Override
protected GuavaCache getCache() {
return cache;
}
@Override
protected Object getNativeCache() {
return nativeCache;
}
@Test
public void putIfAbsentNullValue() throws Exception {
GuavaCache cache = getCache();
Object key = new Object();
Object value = null;
assertNull(cache.get(key));
assertNull(cache.putIfAbsent(key, value));
assertEquals(value, cache.get(key).get());
Cache.ValueWrapper wrapper = cache.putIfAbsent(key, "anotherValue");
assertNotNull(wrapper); // A value is set but is 'null'
assertEquals(null, wrapper.get());
assertEquals(value, cache.get(key).get()); // not changed
}
}

View File

@ -90,6 +90,18 @@ public class TransactionAwareCacheDecoratorTests {
assertEquals("123", target.get(key, String.class)); assertEquals("123", target.get(key, String.class));
} }
@Test
public void putIfAbsent() { // no transactional support for putIfAbsent
Cache target = new ConcurrentMapCache("testCache");
Cache cache = new TransactionAwareCacheDecorator(target);
Object key = new Object();
assertNull(cache.putIfAbsent(key, "123"));
assertEquals("123", target.get(key, String.class));
assertEquals("123", cache.putIfAbsent(key, "456").get());
assertEquals("123", target.get(key, String.class)); // unchanged
}
@Test @Test
public void evictNonTransactional() { public void evictNonTransactional() {
Cache target = new ConcurrentMapCache("testCache"); Cache target = new ConcurrentMapCache("testCache");

View File

@ -80,6 +80,33 @@ public interface Cache {
*/ */
void put(Object key, Object value); void put(Object key, Object value);
/**
* Atomically associate the specified value with the specified key in this cache if
* it is not set already.
* <p>This is equivalent to:
* <pre><code>
* Object existingValue = cache.get(key);
* if (existingValue == null) {
* cache.put(key, value);
* return null;
* } else {
* return existingValue;
* }
* </code></pre>
* except that the action is performed atomically. While all known providers are
* able to perform the put atomically, the returned value may be retrieved after
* the attempt to put (i.e. in a non atomic way). Check the documentation of
* the native cache implementation that you are using for more details.
* @param key the key with which the specified value is to be associated
* @param value the value to be associated with the specified key
* @return the value to which this cache maps the specified key (which may
* be {@code null} itself), or also {@code null} if the cache did not contain
* any mapping for that key prior to this call. Returning {@code null} is
* therefore an indicator that the given {@code value} has been associated
* with the key
*/
ValueWrapper putIfAbsent(Object key, Object value);
/** /**
* Evict the mapping for this key from this cache if it is present. * Evict the mapping for this key from this cache if it is present.
* @param key the key whose mapping is to be removed from the cache * @param key the key whose mapping is to be removed from the cache

View File

@ -103,7 +103,7 @@ public class ConcurrentMapCache implements Cache {
@Override @Override
public ValueWrapper get(Object key) { public ValueWrapper get(Object key) {
Object value = this.store.get(key); Object value = this.store.get(key);
return (value != null ? new SimpleValueWrapper(fromStoreValue(value)) : null); return toWrapper(value);
} }
@Override @Override
@ -121,6 +121,12 @@ public class ConcurrentMapCache implements Cache {
this.store.put(key, toStoreValue(value)); this.store.put(key, toStoreValue(value));
} }
@Override
public ValueWrapper putIfAbsent(Object key, Object value) {
Object existing = this.store.putIfAbsent(key, value);
return toWrapper(existing);
}
@Override @Override
public void evict(Object key) { public void evict(Object key) {
this.store.remove(key); this.store.remove(key);
@ -158,6 +164,9 @@ public class ConcurrentMapCache implements Cache {
return userValue; return userValue;
} }
private ValueWrapper toWrapper(Object value) {
return (value != null ? new SimpleValueWrapper(fromStoreValue(value)) : null);
}
@SuppressWarnings("serial") @SuppressWarnings("serial")
private static class NullHolder implements Serializable { private static class NullHolder implements Serializable {

View File

@ -1,5 +1,5 @@
/* /*
* Copyright 2002-2013 the original author or authors. * Copyright 2002-2014 the original author or authors.
* *
* Licensed under the Apache License, Version 2.0 (the "License"); * Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License. * you may not use this file except in compliance with the License.
@ -34,6 +34,7 @@ import org.springframework.cache.CacheManager;
* <p>Will simply accept any items into the cache not actually storing them. * <p>Will simply accept any items into the cache not actually storing them.
* *
* @author Costin Leau * @author Costin Leau
* @author Stephane Nicoll
* @since 3.1 * @since 3.1
* @see CompositeCacheManager * @see CompositeCacheManager
*/ */
@ -111,6 +112,11 @@ public class NoOpCacheManager implements CacheManager {
@Override @Override
public void put(Object key, Object value) { public void put(Object key, Object value) {
} }
@Override
public ValueWrapper putIfAbsent(Object key, Object value) {
return null;
}
} }
} }

View File

@ -1,5 +1,5 @@
/* /*
* Copyright 2010-2014 the original author or authors. * Copyright 2002-2014 the original author or authors.
* *
* Licensed under the Apache License, Version 2.0 (the "License"); * Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License. * you may not use this file except in compliance with the License.
@ -29,6 +29,7 @@ import static org.junit.Assert.*;
/** /**
* @author Costin Leau * @author Costin Leau
* @author Juergen Hoeller * @author Juergen Hoeller
* @author Stephane Nicoll
*/ */
public class ConcurrentCacheTests { public class ConcurrentCacheTests {
@ -79,6 +80,18 @@ public class ConcurrentCacheTests {
assertNull(cache.get(key, Object.class)); assertNull(cache.get(key, Object.class));
} }
@Test
public void testCachePutIfAbsent() throws Exception {
Object key = new Object();
Object value = "initialValue";
assertNull(cache.get(key));
assertNull(cache.putIfAbsent(key, value));
assertEquals(value, cache.get(key).get());
assertEquals("initialValue", cache.putIfAbsent(key, "anotherValue").get());
assertEquals(value, cache.get(key).get()); // not changed
}
@Test @Test
public void testCacheRemove() throws Exception { public void testCacheRemove() throws Exception {
Object key = "enescu"; Object key = "enescu";