AnnotationUtils.java 22.9 KB
Newer Older
A
Arjen Poutsma 已提交
1
/*
S
Sam Brannen 已提交
2
 * Copyright 2002-2013 the original author or authors.
A
Arjen Poutsma 已提交
3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19
 *
 * 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.core.annotation;

import java.lang.annotation.Annotation;
20
import java.lang.reflect.AnnotatedElement;
A
Arjen Poutsma 已提交
21
import java.lang.reflect.Method;
22
import java.util.List;
A
Arjen Poutsma 已提交
23
import java.util.Map;
24
import java.util.WeakHashMap;
A
Arjen Poutsma 已提交
25 26 27 28 29

import org.springframework.core.BridgeMethodResolver;
import org.springframework.util.Assert;

/**
J
Juergen Hoeller 已提交
30 31 32
 * General utility methods for working with annotations, handling bridge methods (which the compiler
 * generates for generic declarations) as well as super methods (for optional "annotation inheritance").
 * Note that none of this is provided by the JDK's introspection facilities themselves.
A
Arjen Poutsma 已提交
33
 *
J
Juergen Hoeller 已提交
34 35 36 37 38 39
 * <p>As a general rule for runtime-retained annotations (e.g. for transaction control, authorization or service
 * exposure), always use the lookup methods on this class (e.g., {@link #findAnnotation(Method, Class)}, {@link
 * #getAnnotation(Method, Class)}, and {@link #getAnnotations(Method)}) instead of the plain annotation lookup
 * methods in the JDK. You can still explicitly choose between lookup on the given class level only ({@link
 * #getAnnotation(Method, Class)}) and lookup in the entire inheritance hierarchy of the given method ({@link
 * #findAnnotation(Method, Class)}).
A
Arjen Poutsma 已提交
40 41 42 43 44
 *
 * @author Rob Harrop
 * @author Juergen Hoeller
 * @author Sam Brannen
 * @author Mark Fisher
C
Chris Beams 已提交
45
 * @author Chris Beams
A
Arjen Poutsma 已提交
46 47 48 49 50 51 52
 * @since 2.0
 * @see java.lang.reflect.Method#getAnnotations()
 * @see java.lang.reflect.Method#getAnnotation(Class)
 */
public abstract class AnnotationUtils {

	/** The attribute name for annotations with a single element */
J
Juergen Hoeller 已提交
53
	static final String VALUE = "value";
A
Arjen Poutsma 已提交
54

55
	private static final Map<Class<?>, Boolean> annotatedInterfaceCache = new WeakHashMap<Class<?>, Boolean>();
56

J
Juergen Hoeller 已提交
57

58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73
	/**
	 * Get a single {@link Annotation} of {@code annotationType} from the supplied
	 * annotation: either the given annotation itself or a meta-annotation thereof.
	 * @param ann the Annotation to check
	 * @param annotationType the annotation class to look for, both locally and as a meta-annotation
	 * @return the matching annotation or {@code null} if not found
	 * @since 4.0
	 */
	@SuppressWarnings("unchecked")
	public static <T extends Annotation> T getAnnotation(Annotation ann, Class<T> annotationType) {
		if (annotationType.isInstance(ann)) {
			return (T) ann;
		}
		return ann.annotationType().getAnnotation(annotationType);
	}

74
	/**
C
Chris Beams 已提交
75 76 77 78 79 80 81
	 * Get a single {@link Annotation} of {@code annotationType} from the supplied
	 * Method, Constructor or Field. Meta-annotations will be searched if the annotation
	 * is not declared locally on the supplied element.
	 * @param ae the Method, Constructor or Field from which to get the annotation
	 * @param annotationType the annotation class to look for, both locally and as a meta-annotation
	 * @return the matching annotation or {@code null} if not found
	 * @since 3.1
82 83 84 85 86 87 88 89 90 91 92 93 94 95
	 */
	public static <T extends Annotation> T getAnnotation(AnnotatedElement ae, Class<T> annotationType) {
		T ann = ae.getAnnotation(annotationType);
		if (ann == null) {
			for (Annotation metaAnn : ae.getAnnotations()) {
				ann = metaAnn.annotationType().getAnnotation(annotationType);
				if (ann != null) {
					break;
				}
			}
		}
		return ann;
	}

A
Arjen Poutsma 已提交
96 97 98 99 100 101 102 103 104 105 106 107
	/**
	 * Get all {@link Annotation Annotations} from the supplied {@link Method}.
	 * <p>Correctly handles bridge {@link Method Methods} generated by the compiler.
	 * @param method the method to look for annotations on
	 * @return the annotations found
	 * @see org.springframework.core.BridgeMethodResolver#findBridgedMethod(Method)
	 */
	public static Annotation[] getAnnotations(Method method) {
		return BridgeMethodResolver.findBridgedMethod(method).getAnnotations();
	}

	/**
108
	 * Get a single {@link Annotation} of {@code annotationType} from the supplied {@link Method}.
A
Arjen Poutsma 已提交
109 110 111 112 113 114 115
	 * <p>Correctly handles bridge {@link Method Methods} generated by the compiler.
	 * @param method the method to look for annotations on
	 * @param annotationType the annotation class to look for
	 * @return the annotations found
	 * @see org.springframework.core.BridgeMethodResolver#findBridgedMethod(Method)
	 */
	public static <A extends Annotation> A getAnnotation(Method method, Class<A> annotationType) {
116
		Method resolvedMethod = BridgeMethodResolver.findBridgedMethod(method);
117
		return getAnnotation((AnnotatedElement) resolvedMethod, annotationType);
A
Arjen Poutsma 已提交
118 119 120
	}

	/**
121
	 * Get a single {@link Annotation} of {@code annotationType} from the supplied {@link Method},
J
Juergen Hoeller 已提交
122 123
	 * traversing its super methods if no annotation can be found on the given method itself.
	 * <p>Annotations on methods are not inherited by default, so we need to handle this explicitly.
A
Arjen Poutsma 已提交
124 125
	 * @param method the method to look for annotations on
	 * @param annotationType the annotation class to look for
126
	 * @return the annotation found, or {@code null} if none found
A
Arjen Poutsma 已提交
127 128 129
	 */
	public static <A extends Annotation> A findAnnotation(Method method, Class<A> annotationType) {
		A annotation = getAnnotation(method, annotationType);
130
		Class<?> clazz = method.getDeclaringClass();
J
Juergen Hoeller 已提交
131
		if (annotation == null) {
132
			annotation = searchOnInterfaces(method, annotationType, clazz.getInterfaces());
133
		}
A
Arjen Poutsma 已提交
134
		while (annotation == null) {
135 136
			clazz = clazz.getSuperclass();
			if (clazz == null || clazz.equals(Object.class)) {
A
Arjen Poutsma 已提交
137 138 139
				break;
			}
			try {
140
				Method equivalentMethod = clazz.getDeclaredMethod(method.getName(), method.getParameterTypes());
A
Arjen Poutsma 已提交
141 142 143
				annotation = getAnnotation(equivalentMethod, annotationType);
			}
			catch (NoSuchMethodException ex) {
144 145 146
				// No equivalent method found
			}
			if (annotation == null) {
147
				annotation = searchOnInterfaces(method, annotationType, clazz.getInterfaces());
A
Arjen Poutsma 已提交
148 149 150 151 152
			}
		}
		return annotation;
	}

153
	private static <A extends Annotation> A searchOnInterfaces(Method method, Class<A> annotationType, Class<?>[] ifcs) {
154
		A annotation = null;
J
Juergen Hoeller 已提交
155
		for (Class<?> iface : ifcs) {
156 157 158 159 160 161 162 163 164 165 166
			if (isInterfaceWithAnnotatedMethods(iface)) {
				try {
					Method equivalentMethod = iface.getMethod(method.getName(), method.getParameterTypes());
					annotation = getAnnotation(equivalentMethod, annotationType);
				}
				catch (NoSuchMethodException ex) {
					// Skip this interface - it doesn't have the method...
				}
				if (annotation != null) {
					break;
				}
167
			}
168 169 170 171 172 173 174 175 176
		}
		return annotation;
	}

	private static boolean isInterfaceWithAnnotatedMethods(Class<?> iface) {
		synchronized (annotatedInterfaceCache) {
			Boolean flag = annotatedInterfaceCache.get(iface);
			if (flag != null) {
				return flag;
177
			}
178 179 180 181 182 183
			boolean found = false;
			for (Method ifcMethod : iface.getMethods()) {
				if (ifcMethod.getAnnotations().length > 0) {
					found = true;
					break;
				}
184
			}
185 186
			annotatedInterfaceCache.put(iface, found);
			return found;
187 188 189
		}
	}

A
Arjen Poutsma 已提交
190
	/**
191
	 * Find a single {@link Annotation} of {@code annotationType} from the supplied {@link Class},
192
	 * traversing its interfaces and superclasses if no annotation can be found on the given class itself.
J
Juergen Hoeller 已提交
193
	 * <p>This method explicitly handles class-level annotations which are not declared as
194
	 * {@link java.lang.annotation.Inherited inherited} <i>as well as annotations on interfaces</i>.
J
Juergen Hoeller 已提交
195 196 197 198 199 200
	 * <p>The algorithm operates as follows: Searches for an annotation on the given class and returns
	 * it if found. Else searches all interfaces that the given class declares, returning the annotation
	 * from the first matching candidate, if any. Else proceeds with introspection of the superclass
	 * of the given class, checking the superclass itself; if no annotation found there, proceeds
	 * with the interfaces that the superclass declares. Recursing up through the entire superclass
	 * hierarchy if no match is found.
A
Arjen Poutsma 已提交
201 202
	 * @param clazz the class to look for annotations on
	 * @param annotationType the annotation class to look for
203
	 * @return the annotation found, or {@code null} if none found
A
Arjen Poutsma 已提交
204 205 206 207 208 209 210 211 212 213 214 215 216
	 */
	public static <A extends Annotation> A findAnnotation(Class<?> clazz, Class<A> annotationType) {
		Assert.notNull(clazz, "Class must not be null");
		A annotation = clazz.getAnnotation(annotationType);
		if (annotation != null) {
			return annotation;
		}
		for (Class<?> ifc : clazz.getInterfaces()) {
			annotation = findAnnotation(ifc, annotationType);
			if (annotation != null) {
				return annotation;
			}
		}
217 218 219 220 221 222 223 224
		if (!Annotation.class.isAssignableFrom(clazz)) {
			for (Annotation ann : clazz.getAnnotations()) {
				annotation = findAnnotation(ann.annotationType(), annotationType);
				if (annotation != null) {
					return annotation;
				}
			}
		}
225
		Class<?> superClass = clazz.getSuperclass();
226
		if (superClass == null || superClass.equals(Object.class)) {
A
Arjen Poutsma 已提交
227 228 229 230 231 232
			return null;
		}
		return findAnnotation(superClass, annotationType);
	}

	/**
233 234 235 236 237
	 * Find the first {@link Class} in the inheritance hierarchy of the specified {@code clazz}
	 * (including the specified {@code clazz} itself) which declares an annotation for the
	 * specified {@code annotationType}, or {@code null} if not found. If the supplied
	 * {@code clazz} is {@code null}, {@code null} will be returned.
	 * <p>If the supplied {@code clazz} is an interface, only the interface itself will be checked;
J
Juergen Hoeller 已提交
238 239 240 241
	 * the inheritance hierarchy for interfaces will not be traversed.
	 * <p>The standard {@link Class} API does not provide a mechanism for determining which class
	 * in an inheritance hierarchy actually declares an {@link Annotation}, so we need to handle
	 * this explicitly.
A
Arjen Poutsma 已提交
242
	 * @param annotationType the Class object corresponding to the annotation type
J
Juergen Hoeller 已提交
243
	 * @param clazz the Class object corresponding to the class on which to check for the annotation,
244 245 246
	 * or {@code null}
	 * @return the first {@link Class} in the inheritance hierarchy of the specified {@code clazz}
	 * which declares an annotation for the specified {@code annotationType}, or {@code null}
J
Juergen Hoeller 已提交
247
	 * if not found
A
Arjen Poutsma 已提交
248 249
	 * @see Class#isAnnotationPresent(Class)
	 * @see Class#getDeclaredAnnotations()
250 251
	 * @see #findAnnotationDeclaringClassForTypes(List, Class)
	 * @see #isAnnotationDeclaredLocally(Class, Class)
A
Arjen Poutsma 已提交
252 253 254 255 256 257
	 */
	public static Class<?> findAnnotationDeclaringClass(Class<? extends Annotation> annotationType, Class<?> clazz) {
		Assert.notNull(annotationType, "Annotation type must not be null");
		if (clazz == null || clazz.equals(Object.class)) {
			return null;
		}
258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281
		return (isAnnotationDeclaredLocally(annotationType, clazz)) ? clazz : findAnnotationDeclaringClass(
			annotationType, clazz.getSuperclass());
	}

	/**
	 * Find the first {@link Class} in the inheritance hierarchy of the specified
	 * {@code clazz} (including the specified {@code clazz} itself) which declares
	 * at least one of the specified {@code annotationTypes}, or {@code null} if
	 * none of the specified annotation types could be found.
	 * <p>If the supplied {@code clazz} is {@code null}, {@code null} will be
	 * returned.
	 * <p>If the supplied {@code clazz} is an interface, only the interface itself
	 * will be checked; the inheritance hierarchy for interfaces will not be traversed.
	 * <p>The standard {@link Class} API does not provide a mechanism for determining
	 * which class in an inheritance hierarchy actually declares one of several
	 * candidate {@linkplain Annotation annotations}, so we need to handle this
	 * explicitly.
	 * @param annotationTypes the list of Class objects corresponding to the
	 * annotation types
	 * @param clazz the Class object corresponding to the class on which to check
	 * for the annotations, or {@code null}
	 * @return the first {@link Class} in the inheritance hierarchy of the specified
	 * {@code clazz} which declares an annotation of at least one of the specified
	 * {@code annotationTypes}, or {@code null} if not found
282
	 * @since 3.2.2
283 284 285 286 287
	 * @see Class#isAnnotationPresent(Class)
	 * @see Class#getDeclaredAnnotations()
	 * @see #findAnnotationDeclaringClass(Class, Class)
	 * @see #isAnnotationDeclaredLocally(Class, Class)
	 */
288
	public static Class<?> findAnnotationDeclaringClassForTypes(List<Class<? extends Annotation>> annotationTypes, Class<?> clazz) {
289 290 291 292 293 294 295 296 297 298
		Assert.notEmpty(annotationTypes, "The list of annotation types must not be empty");
		if (clazz == null || clazz.equals(Object.class)) {
			return null;
		}
		for (Class<? extends Annotation> annotationType : annotationTypes) {
			if (isAnnotationDeclaredLocally(annotationType, clazz)) {
				return clazz;
			}
		}
		return findAnnotationDeclaringClassForTypes(annotationTypes, clazz.getSuperclass());
A
Arjen Poutsma 已提交
299 300 301
	}

	/**
302 303
	 * Determine whether an annotation for the specified {@code annotationType} is
	 * declared locally on the supplied {@code clazz}. The supplied {@link Class}
J
Juergen Hoeller 已提交
304 305
	 * may represent any type.
	 * <p>Note: This method does <strong>not</strong> determine if the annotation is
S
Sam Brannen 已提交
306 307 308
	 * {@linkplain java.lang.annotation.Inherited inherited}. For greater clarity
	 * regarding inherited annotations, consider using
	 * {@link #isAnnotationInherited(Class, Class)} instead.
A
Arjen Poutsma 已提交
309
	 * @param annotationType the Class object corresponding to the annotation type
J
Juergen Hoeller 已提交
310
	 * @param clazz the Class object corresponding to the class on which to check for the annotation
311 312
	 * @return {@code true} if an annotation for the specified {@code annotationType}
	 * is declared locally on the supplied {@code clazz}
A
Arjen Poutsma 已提交
313 314 315 316 317 318 319
	 * @see Class#getDeclaredAnnotations()
	 * @see #isAnnotationInherited(Class, Class)
	 */
	public static boolean isAnnotationDeclaredLocally(Class<? extends Annotation> annotationType, Class<?> clazz) {
		Assert.notNull(annotationType, "Annotation type must not be null");
		Assert.notNull(clazz, "Class must not be null");
		boolean declaredLocally = false;
S
Sam Brannen 已提交
320
		for (Annotation annotation : clazz.getDeclaredAnnotations()) {
A
Arjen Poutsma 已提交
321 322 323 324 325 326 327 328 329
			if (annotation.annotationType().equals(annotationType)) {
				declaredLocally = true;
				break;
			}
		}
		return declaredLocally;
	}

	/**
330
	 * Determine whether an annotation for the specified {@code annotationType} is present
S
Sam Brannen 已提交
331 332
	 * on the supplied {@code clazz} and is {@linkplain java.lang.annotation.Inherited inherited}
	 * (i.e., not declared locally for the class).
333
	 * <p>If the supplied {@code clazz} is an interface, only the interface itself will be checked.
J
Juergen Hoeller 已提交
334
	 * In accordance with standard meta-annotation semantics, the inheritance hierarchy for interfaces
S
Sam Brannen 已提交
335 336
	 * will not be traversed. See the {@linkplain java.lang.annotation.Inherited Javadoc} for the
	 * {@code @Inherited} meta-annotation for further details regarding annotation inheritance.
A
Arjen Poutsma 已提交
337
	 * @param annotationType the Class object corresponding to the annotation type
J
Juergen Hoeller 已提交
338
	 * @param clazz the Class object corresponding to the class on which to check for the annotation
339
	 * @return {@code true} if an annotation for the specified {@code annotationType} is present
S
Sam Brannen 已提交
340
	 * on the supplied {@code clazz} and is <em>inherited</em>
A
Arjen Poutsma 已提交
341 342 343 344 345 346 347 348 349 350
	 * @see Class#isAnnotationPresent(Class)
	 * @see #isAnnotationDeclaredLocally(Class, Class)
	 */
	public static boolean isAnnotationInherited(Class<? extends Annotation> annotationType, Class<?> clazz) {
		Assert.notNull(annotationType, "Annotation type must not be null");
		Assert.notNull(clazz, "Class must not be null");
		return (clazz.isAnnotationPresent(annotationType) && !isAnnotationDeclaredLocally(annotationType, clazz));
	}

	/**
351 352 353 354 355
	 * Retrieve the given annotation's attributes as a Map, preserving all attribute types
	 * as-is.
	 * <p>Note: As of Spring 3.1.1, the returned map is actually an
	 * {@link AnnotationAttributes} instance, however the Map signature of this method has
	 * been preserved for binary compatibility.
A
Arjen Poutsma 已提交
356
	 * @param annotation the annotation to retrieve the attributes for
J
Juergen Hoeller 已提交
357 358
	 * @return the Map of annotation attributes, with attribute names as keys and
	 * corresponding attribute values as values
A
Arjen Poutsma 已提交
359 360
	 */
	public static Map<String, Object> getAnnotationAttributes(Annotation annotation) {
361
		return getAnnotationAttributes(annotation, false, false);
362 363 364
	}

	/**
365 366 367 368 369 370
	 * Retrieve the given annotation's attributes as a Map. Equivalent to calling
	 * {@link #getAnnotationAttributes(Annotation, boolean, boolean)} with
	 * the {@code nestedAnnotationsAsMap} parameter set to {@code false}.
	 * <p>Note: As of Spring 3.1.1, the returned map is actually an
	 * {@link AnnotationAttributes} instance, however the Map signature of this method has
	 * been preserved for binary compatibility.
371
	 * @param annotation the annotation to retrieve the attributes for
372 373 374
	 * @param classValuesAsString whether to turn Class references into Strings (for
	 * compatibility with {@link org.springframework.core.type.AnnotationMetadata} or to
	 * preserve them as Class references
J
Juergen Hoeller 已提交
375 376
	 * @return the Map of annotation attributes, with attribute names as keys and
	 * corresponding attribute values as values
377
	 */
378
	public static Map<String, Object> getAnnotationAttributes(Annotation annotation, boolean classValuesAsString) {
379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398
		return getAnnotationAttributes(annotation, classValuesAsString, false);
	}

	/**
	 * Retrieve the given annotation's attributes as an {@link AnnotationAttributes}
	 * map structure. Implemented in Spring 3.1.1 to provide fully recursive annotation
	 * reading capabilities on par with that of the reflection-based
	 * {@link org.springframework.core.type.StandardAnnotationMetadata}.
	 * @param annotation the annotation to retrieve the attributes for
	 * @param classValuesAsString whether to turn Class references into Strings (for
	 * compatibility with {@link org.springframework.core.type.AnnotationMetadata} or to
	 * preserve them as Class references
	 * @param nestedAnnotationsAsMap whether to turn nested Annotation instances into
	 * {@link AnnotationAttributes} maps (for compatibility with
	 * {@link org.springframework.core.type.AnnotationMetadata} or to preserve them as
	 * Annotation instances
	 * @return the annotation attributes (a specialized Map) with attribute names as keys
	 * and corresponding attribute values as values
	 * @since 3.1.1
	 */
399 400
	public static AnnotationAttributes getAnnotationAttributes(Annotation annotation, boolean classValuesAsString,
			boolean nestedAnnotationsAsMap) {
401 402

		AnnotationAttributes attrs = new AnnotationAttributes();
A
Arjen Poutsma 已提交
403
		Method[] methods = annotation.annotationType().getDeclaredMethods();
404
		for (Method method : methods) {
A
Arjen Poutsma 已提交
405 406
			if (method.getParameterTypes().length == 0 && method.getReturnType() != void.class) {
				try {
407
					Object value = method.invoke(annotation);
408
					if (classValuesAsString) {
409
						if (value instanceof Class) {
410
							value = ((Class<?>) value).getName();
411 412
						}
						else if (value instanceof Class[]) {
413
							Class<?>[] clazzArray = (Class[]) value;
414 415 416 417 418 419 420
							String[] newValue = new String[clazzArray.length];
							for (int i = 0; i < clazzArray.length; i++) {
								newValue[i] = clazzArray[i].getName();
							}
							value = newValue;
						}
					}
421
					if (nestedAnnotationsAsMap && value instanceof Annotation) {
422 423
						attrs.put(method.getName(),
							getAnnotationAttributes((Annotation) value, classValuesAsString, nestedAnnotationsAsMap));
424 425
					}
					else if (nestedAnnotationsAsMap && value instanceof Annotation[]) {
426
						Annotation[] realAnnotations = (Annotation[]) value;
427 428
						AnnotationAttributes[] mappedAnnotations = new AnnotationAttributes[realAnnotations.length];
						for (int i = 0; i < realAnnotations.length; i++) {
429 430
							mappedAnnotations[i] = getAnnotationAttributes(
									realAnnotations[i], classValuesAsString, nestedAnnotationsAsMap);
431 432 433 434 435 436
						}
						attrs.put(method.getName(), mappedAnnotations);
					}
					else {
						attrs.put(method.getName(), value);
					}
A
Arjen Poutsma 已提交
437 438 439 440 441 442 443 444 445 446
				}
				catch (Exception ex) {
					throw new IllegalStateException("Could not obtain annotation attribute values", ex);
				}
			}
		}
		return attrs;
	}

	/**
447
	 * Retrieve the <em>value</em> of the {@code &quot;value&quot;} attribute of a
J
Juergen Hoeller 已提交
448
	 * single-element Annotation, given an annotation instance.
A
Arjen Poutsma 已提交
449
	 * @param annotation the annotation instance from which to retrieve the value
450
	 * @return the attribute value, or {@code null} if not found
A
Arjen Poutsma 已提交
451 452 453 454 455 456 457
	 * @see #getValue(Annotation, String)
	 */
	public static Object getValue(Annotation annotation) {
		return getValue(annotation, VALUE);
	}

	/**
J
Juergen Hoeller 已提交
458
	 * Retrieve the <em>value</em> of a named Annotation attribute, given an annotation instance.
A
Arjen Poutsma 已提交
459 460
	 * @param annotation the annotation instance from which to retrieve the value
	 * @param attributeName the name of the attribute value to retrieve
461
	 * @return the attribute value, or {@code null} if not found
J
Juergen Hoeller 已提交
462
	 * @see #getValue(Annotation)
A
Arjen Poutsma 已提交
463 464 465 466 467 468 469 470 471 472 473 474
	 */
	public static Object getValue(Annotation annotation, String attributeName) {
		try {
			Method method = annotation.annotationType().getDeclaredMethod(attributeName, new Class[0]);
			return method.invoke(annotation);
		}
		catch (Exception ex) {
			return null;
		}
	}

	/**
475
	 * Retrieve the <em>default value</em> of the {@code &quot;value&quot;} attribute
J
Juergen Hoeller 已提交
476 477
	 * of a single-element Annotation, given an annotation instance.
	 * @param annotation the annotation instance from which to retrieve the default value
478
	 * @return the default value, or {@code null} if not found
A
Arjen Poutsma 已提交
479 480 481 482 483 484 485
	 * @see #getDefaultValue(Annotation, String)
	 */
	public static Object getDefaultValue(Annotation annotation) {
		return getDefaultValue(annotation, VALUE);
	}

	/**
J
Juergen Hoeller 已提交
486 487
	 * Retrieve the <em>default value</em> of a named Annotation attribute, given an annotation instance.
	 * @param annotation the annotation instance from which to retrieve the default value
A
Arjen Poutsma 已提交
488
	 * @param attributeName the name of the attribute value to retrieve
489
	 * @return the default value of the named attribute, or {@code null} if not found
A
Arjen Poutsma 已提交
490 491 492 493 494 495 496
	 * @see #getDefaultValue(Class, String)
	 */
	public static Object getDefaultValue(Annotation annotation, String attributeName) {
		return getDefaultValue(annotation.annotationType(), attributeName);
	}

	/**
497
	 * Retrieve the <em>default value</em> of the {@code &quot;value&quot;} attribute
J
Juergen Hoeller 已提交
498 499
	 * of a single-element Annotation, given the {@link Class annotation type}.
	 * @param annotationType the <em>annotation type</em> for which the default value should be retrieved
500
	 * @return the default value, or {@code null} if not found
A
Arjen Poutsma 已提交
501 502 503 504 505 506 507
	 * @see #getDefaultValue(Class, String)
	 */
	public static Object getDefaultValue(Class<? extends Annotation> annotationType) {
		return getDefaultValue(annotationType, VALUE);
	}

	/**
J
Juergen Hoeller 已提交
508 509
	 * Retrieve the <em>default value</em> of a named Annotation attribute, given the {@link Class annotation type}.
	 * @param annotationType the <em>annotation type</em> for which the default value should be retrieved
A
Arjen Poutsma 已提交
510
	 * @param attributeName the name of the attribute value to retrieve.
511
	 * @return the default value of the named attribute, or {@code null} if not found
A
Arjen Poutsma 已提交
512 513 514 515 516 517 518 519 520 521 522 523 524
	 * @see #getDefaultValue(Annotation, String)
	 */
	public static Object getDefaultValue(Class<? extends Annotation> annotationType, String attributeName) {
		try {
			Method method = annotationType.getDeclaredMethod(attributeName, new Class[0]);
			return method.getDefaultValue();
		}
		catch (Exception ex) {
			return null;
		}
	}

}