Lazily retrieve TypeDescriptor annotations on demand

Closes gh-33948
This commit is contained in:
Juergen Hoeller 2025-02-07 18:55:24 +01:00
parent bb7a8006c5
commit 1a573d6e3c
1 changed files with 30 additions and 11 deletions

View File

@ -1,5 +1,5 @@
/* /*
* Copyright 2002-2024 the original author or authors. * Copyright 2002-2025 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.
@ -25,6 +25,7 @@ import java.util.Arrays;
import java.util.Collection; import java.util.Collection;
import java.util.HashMap; import java.util.HashMap;
import java.util.Map; import java.util.Map;
import java.util.function.Supplier;
import java.util.stream.Stream; import java.util.stream.Stream;
import org.springframework.core.MethodParameter; import org.springframework.core.MethodParameter;
@ -70,7 +71,10 @@ public class TypeDescriptor implements Serializable {
private final ResolvableType resolvableType; private final ResolvableType resolvableType;
private final AnnotatedElementAdapter annotatedElement; private final AnnotatedElementSupplier annotatedElementSupplier;
@Nullable
private volatile AnnotatedElementAdapter annotatedElement;
/** /**
@ -82,7 +86,7 @@ public class TypeDescriptor implements Serializable {
public TypeDescriptor(MethodParameter methodParameter) { public TypeDescriptor(MethodParameter methodParameter) {
this.resolvableType = ResolvableType.forMethodParameter(methodParameter); this.resolvableType = ResolvableType.forMethodParameter(methodParameter);
this.type = this.resolvableType.resolve(methodParameter.getNestedParameterType()); this.type = this.resolvableType.resolve(methodParameter.getNestedParameterType());
this.annotatedElement = AnnotatedElementAdapter.from(methodParameter.getParameterIndex() == -1 ? this.annotatedElementSupplier = () -> AnnotatedElementAdapter.from(methodParameter.getParameterIndex() == -1 ?
methodParameter.getMethodAnnotations() : methodParameter.getParameterAnnotations()); methodParameter.getMethodAnnotations() : methodParameter.getParameterAnnotations());
} }
@ -94,7 +98,7 @@ public class TypeDescriptor implements Serializable {
public TypeDescriptor(Field field) { public TypeDescriptor(Field field) {
this.resolvableType = ResolvableType.forField(field); this.resolvableType = ResolvableType.forField(field);
this.type = this.resolvableType.resolve(field.getType()); this.type = this.resolvableType.resolve(field.getType());
this.annotatedElement = AnnotatedElementAdapter.from(field.getAnnotations()); this.annotatedElementSupplier = () -> AnnotatedElementAdapter.from(field.getAnnotations());
} }
/** /**
@ -107,7 +111,7 @@ public class TypeDescriptor implements Serializable {
Assert.notNull(property, "Property must not be null"); Assert.notNull(property, "Property must not be null");
this.resolvableType = ResolvableType.forMethodParameter(property.getMethodParameter()); this.resolvableType = ResolvableType.forMethodParameter(property.getMethodParameter());
this.type = this.resolvableType.resolve(property.getType()); this.type = this.resolvableType.resolve(property.getType());
this.annotatedElement = AnnotatedElementAdapter.from(property.getAnnotations()); this.annotatedElementSupplier = () -> AnnotatedElementAdapter.from(property.getAnnotations());
} }
/** /**
@ -123,7 +127,7 @@ public class TypeDescriptor implements Serializable {
public TypeDescriptor(ResolvableType resolvableType, @Nullable Class<?> type, @Nullable Annotation[] annotations) { public TypeDescriptor(ResolvableType resolvableType, @Nullable Class<?> type, @Nullable Annotation[] annotations) {
this.resolvableType = resolvableType; this.resolvableType = resolvableType;
this.type = (type != null ? type : resolvableType.toClass()); this.type = (type != null ? type : resolvableType.toClass());
this.annotatedElement = AnnotatedElementAdapter.from(annotations); this.annotatedElementSupplier = () -> AnnotatedElementAdapter.from(annotations);
} }
@ -250,12 +254,21 @@ public class TypeDescriptor implements Serializable {
return getType().isPrimitive(); return getType().isPrimitive();
} }
private AnnotatedElementAdapter getAnnotatedElement() {
AnnotatedElementAdapter annotatedElement = this.annotatedElement;
if (annotatedElement == null) {
annotatedElement = this.annotatedElementSupplier.get();
this.annotatedElement = annotatedElement;
}
return annotatedElement;
}
/** /**
* Return the annotations associated with this type descriptor, if any. * Return the annotations associated with this type descriptor, if any.
* @return the annotations, or an empty array if none * @return the annotations, or an empty array if none
*/ */
public Annotation[] getAnnotations() { public Annotation[] getAnnotations() {
return this.annotatedElement.getAnnotations(); return getAnnotatedElement().getAnnotations();
} }
/** /**
@ -266,12 +279,13 @@ public class TypeDescriptor implements Serializable {
* @return {@code true} if the annotation is present * @return {@code true} if the annotation is present
*/ */
public boolean hasAnnotation(Class<? extends Annotation> annotationType) { public boolean hasAnnotation(Class<? extends Annotation> annotationType) {
if (this.annotatedElement.isEmpty()) { AnnotatedElementAdapter annotatedElement = getAnnotatedElement();
if (annotatedElement.isEmpty()) {
// Shortcut: AnnotatedElementUtils would have to expect AnnotatedElement.getAnnotations() // Shortcut: AnnotatedElementUtils would have to expect AnnotatedElement.getAnnotations()
// to return a copy of the array, whereas we can do it more efficiently here. // to return a copy of the array, whereas we can do it more efficiently here.
return false; return false;
} }
return AnnotatedElementUtils.isAnnotated(this.annotatedElement, annotationType); return AnnotatedElementUtils.isAnnotated(annotatedElement, annotationType);
} }
/** /**
@ -282,12 +296,13 @@ public class TypeDescriptor implements Serializable {
*/ */
@Nullable @Nullable
public <T extends Annotation> T getAnnotation(Class<T> annotationType) { public <T extends Annotation> T getAnnotation(Class<T> annotationType) {
if (this.annotatedElement.isEmpty()) { AnnotatedElementAdapter annotatedElement = getAnnotatedElement();
if (annotatedElement.isEmpty()) {
// Shortcut: AnnotatedElementUtils would have to expect AnnotatedElement.getAnnotations() // Shortcut: AnnotatedElementUtils would have to expect AnnotatedElement.getAnnotations()
// to return a copy of the array, whereas we can do it more efficiently here. // to return a copy of the array, whereas we can do it more efficiently here.
return null; return null;
} }
return AnnotatedElementUtils.getMergedAnnotation(this.annotatedElement, annotationType); return AnnotatedElementUtils.getMergedAnnotation(annotatedElement, annotationType);
} }
/** /**
@ -808,4 +823,8 @@ public class TypeDescriptor implements Serializable {
} }
} }
private interface AnnotatedElementSupplier extends Supplier<AnnotatedElementAdapter>, Serializable {
}
} }