I'm trying to implement Spring 3.1 caching as explained here and here, but it doesn't seem to be working: my method is run through every time even though it is marked @cacheable. What am I doing wrong?
I've moved it into a junit test case with its own configuration file to isolate it from the rest of my application, but the problem still happens. Here are the relevant files:
Spring-test-servlet.xml
<beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:cache="http://www.springframework.org/schema/cache" xmlns:p="http://www.springframework.org/schema/p" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/cache http://www.springframework.org/schema/cache/spring-cache.xsd"> <cache:annotation-driven /> <bean id="cacheManager" class="org.springframework.cache.ehcache.EhCacheCacheManager" p:cache-manager-ref="ehcache"/> <bean id="ehcache" class="org.springframework.cache.ehcache.EhCacheManagerFactoryBean" p:config-location="classpath:ehcache.xml"/> </beans>
ehcache.xml
<ehcache> <diskStore path="java.io.tmpdir"/> <cache name="cache" maxElementsInMemory="100" eternal="false" timeToIdleSeconds="120" timeToLiveSeconds="120" overflowToDisk="true" maxElementsOnDisk="10000000" diskPersistent="false" diskExpiryThreadIntervalSeconds="120" memoryStoreEvictionPolicy="LRU"/> </ehcache>
MyTest.java
@RunWith(SpringJUnit4ClassRunner.class) @ContextConfiguration({"classpath:spring-test-servlet.xml"}) @Component public class MyTest extends TestCase { @Test public void testCache1(){ for(int i = 0; i < 5; i++){ System.out.println("Calling someMethod..."); System.out.println(someMethod(0)); } } @Cacheable("testmethod") private int someMethod(int val){ System.out.println("Not from cache"); return 5; } }
Relevant Pom entries: (spring-version = 3.1.1.RELEASE)
<dependency> <groupId>net.sf.ehcache</groupId> <artifactId>ehcache-core</artifactId> <version>2.5.1</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-context-support</artifactId> <version>${spring.version}</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-context</artifactId> <version>${spring.version}</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-core</artifactId> <version>${spring.version}</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-test</artifactId> <version>${spring.version}</version> </dependency>
when I run the test, Spring puts out some debug messages that looks like my cache is initialized without errors
DEBUG: config.ConfigurationHelper - No CacheManagerEventListenerFactory class specified. Skipping... DEBUG: ehcache.Cache - No BootstrapCacheLoaderFactory class specified. Skipping... DEBUG: ehcache.Cache - CacheWriter factory not configured. Skipping... DEBUG: config.ConfigurationHelper - No CacheExceptionHandlerFactory class specified. Skipping... DEBUG: store.MemoryStore - Initialized net.sf.ehcache.store.MemoryStore for cache DEBUG: disk.DiskStorageFactory - Failed to delete file cache.data DEBUG: disk.DiskStorageFactory - Failed to delete file cache.index DEBUG: disk.DiskStorageFactory - Matching data file missing (or empty) for index file. Deleting index file /var/folders/qg/xwdvsg6x3mx_z_rcfvq7lc0m0000gn/T/cache.index DEBUG: disk.DiskStorageFactory - Failed to delete file cache.index DEBUG: ehcache.Cache - Initialised cache: cache DEBUG: config.ConfigurationHelper - CacheDecoratorFactory not configured. Skipping for 'cache'. DEBUG: config.ConfigurationHelper - CacheDecoratorFactory not configured for defaultCache. Skipping for 'cache'.
but the debug output shows no cache checks between method calls to someMethod and the print statement from inside someMethod prints every time.
Is there something I'm missing?
From http://static.springsource.org/spring/docs/3.1.x/spring-framework-reference/html/cache.html
In proxy mode (which is the default), only external method calls coming in through the proxy are intercepted. This means that self-invocation, in effect, a method within the target object calling another method of the target object, will not lead to an actual caching at runtime even if the invoked method is marked with @Cacheable - considering using the aspectj mode in this case.
and
Method visibility and
@Cacheable
/@CachePut
/@CacheEvict
When using proxies, you should apply the
@Cache
annotations only to methods with public visibility.
someMethod
in the same target object. @Cacheable
method is not public.You need to define a cache that matches the name you are referencing in you annotation ("testmethod"). Create an entry in your ehcache.xml for that cache as well.
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With