Java tutorial
/* * Copyright 2002-2011 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 com.icfcc.cache.annotation; import com.icfcc.cache.interceptor.*; import com.icfcc.cache.interceptor.AbstractFallbackCacheOperationSource; import com.icfcc.cache.interceptor.CacheOperation; import org.springframework.util.Assert; import java.io.Serializable; import java.lang.reflect.AnnotatedElement; import java.lang.reflect.Method; import java.util.*; /** * Implementation of the {@link CacheOperationSource * CacheOperationSource} interface for working with caching metadata in annotation format. * * <p>This class reads Spring's {@link Cacheable}, {@link CachePut} and {@link CacheEvict} * annotations and exposes corresponding caching operation definition to Spring's cache * infrastructure. This class may also serve as base class for a custom * {@code CacheOperationSource}. * * @author Costin Leau * @since 3.1 */ @SuppressWarnings("serial") public class AnnotationCacheOperationSource extends AbstractFallbackCacheOperationSource implements Serializable { private final boolean publicMethodsOnly; private final Set<CacheAnnotationParser> annotationParsers; /** * Create a default AnnotationCacheOperationSource, supporting public methods * that carry the {@code Cacheable} and {@code CacheEvict} annotations. */ public AnnotationCacheOperationSource() { this(true); } /** * Create a default {@code AnnotationCacheOperationSource}, supporting public methods * that carry the {@code Cacheable} and {@code CacheEvict} annotations. * @param publicMethodsOnly whether to support only annotated public methods * typically for use with proxy-based AOP), or protected/private methods as well * (typically used with AspectJ class weaving) */ public AnnotationCacheOperationSource(boolean publicMethodsOnly) { this.publicMethodsOnly = publicMethodsOnly; this.annotationParsers = new LinkedHashSet<CacheAnnotationParser>(1); this.annotationParsers.add(new SpringCacheAnnotationParser()); } /** * Create a custom AnnotationCacheOperationSource. * @param annotationParsers the CacheAnnotationParser to use */ public AnnotationCacheOperationSource(CacheAnnotationParser... annotationParsers) { this.publicMethodsOnly = true; Assert.notEmpty(annotationParsers, "At least one CacheAnnotationParser needs to be specified"); Set<CacheAnnotationParser> parsers = new LinkedHashSet<CacheAnnotationParser>(annotationParsers.length); Collections.addAll(parsers, annotationParsers); this.annotationParsers = parsers; } @Override protected Collection<CacheOperation> findCacheOperations(Class<?> clazz) { return determineCacheOperations(clazz); } @Override protected Collection<CacheOperation> findCacheOperations(Method method) { return determineCacheOperations(method); } /** * Determine the cache operation(s) for the given method or class. * <p>This implementation delegates to configured * {@link CacheAnnotationParser}s for parsing known annotations into * Spring's metadata attribute class. * <p>Can be overridden to support custom annotations that carry * caching metadata. * @param ae the annotated method or class * @return the configured caching operations, or {@code null} if none found */ protected Collection<CacheOperation> determineCacheOperations(AnnotatedElement ae) { Collection<CacheOperation> ops = null; for (CacheAnnotationParser annotationParser : this.annotationParsers) { Collection<CacheOperation> annOps = annotationParser.parseCacheAnnotations(ae); if (annOps != null) { if (ops == null) { ops = new ArrayList<CacheOperation>(); } ops.addAll(annOps); } } return ops; } /** * By default, only public methods can be made cacheable. */ @Override protected boolean allowPublicMethodsOnly() { return this.publicMethodsOnly; } }