All Downloads are FREE. Search and download functionalities are using the official Maven repository.

infra.beans.factory.annotation.ParameterResolutionDelegate Maven / Gradle / Ivy

The newest version!
/*
 * Copyright 2017 - 2024 the original author or authors.
 *
 * This program is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program. If not, see [https://www.gnu.org/licenses/]
 */

package infra.beans.factory.annotation;

import java.lang.annotation.Annotation;
import java.lang.reflect.AnnotatedElement;
import java.lang.reflect.Constructor;
import java.lang.reflect.Executable;
import java.lang.reflect.Parameter;

import infra.beans.BeansException;
import infra.beans.factory.config.AutowireCapableBeanFactory;
import infra.beans.factory.config.DependencyDescriptor;
import infra.core.MethodParameter;
import infra.core.annotation.AnnotatedElementUtils;
import infra.core.annotation.SynthesizingMethodParameter;
import infra.lang.Assert;
import infra.lang.Nullable;
import infra.util.ClassUtils;

/**
 * Public delegate for resolving autowirable parameters on externally managed
 * constructors and methods.
 *
 * @author Sam Brannen
 * @author Juergen Hoeller
 * @author Harry Yang
 * @see #isAutowirable
 * @see #resolveDependency
 * @since 4.0 2022/3/8 13:46
 */
public final class ParameterResolutionDelegate {

  private static final AnnotatedElement EMPTY_ANNOTATED_ELEMENT = new AnnotatedElement() {
    @Override
    @Nullable
    public  T getAnnotation(Class annotationClass) {
      return null;
    }

    @Override
    public Annotation[] getAnnotations() {
      return new Annotation[0];
    }

    @Override
    public Annotation[] getDeclaredAnnotations() {
      return new Annotation[0];
    }
  };

  private ParameterResolutionDelegate() { }

  /**
   * Determine if the supplied {@link Parameter} can potentially be
   * autowired from an {@link AutowireCapableBeanFactory}.
   * 

Returns {@code true} if the supplied parameter is annotated or * meta-annotated with {@link Autowired @Autowired}, * {@link Qualifier @Qualifier}, or {@link Value @Value}. *

Note that {@link #resolveDependency} may still be able to resolve the * dependency for the supplied parameter even if this method returns {@code false}. * * @param parameter the parameter whose dependency should be autowired * (must not be {@code null}) * @param parameterIndex the index of the parameter in the constructor or method * that declares the parameter * @see #resolveDependency */ public static boolean isAutowirable(Parameter parameter, int parameterIndex) { Assert.notNull(parameter, "Parameter is required"); AnnotatedElement annotatedParameter = getEffectiveAnnotatedParameter(parameter, parameterIndex); return AnnotatedElementUtils.hasAnnotation(annotatedParameter, Autowired.class) || AnnotatedElementUtils.hasAnnotation(annotatedParameter, Qualifier.class) || AnnotatedElementUtils.hasAnnotation(annotatedParameter, Value.class); } /** * Resolve the dependency for the supplied {@link Parameter} from the * supplied {@link AutowireCapableBeanFactory}. *

Provides comprehensive autowiring support for individual method parameters * on par with Framework's dependency injection facilities for autowired fields and * methods, including support for {@link Autowired @Autowired}, * {@link Qualifier @Qualifier}, and {@link Value @Value} with support for property * placeholders and EL expressions in {@code @Value} declarations. *

The dependency is required unless the parameter is annotated or meta-annotated * with {@link Autowired @Autowired} with the {@link Autowired#required required} * flag set to {@code false}. *

If an explicit qualifier is not declared, the name of the parameter * will be used as the qualifier for resolving ambiguities. * * @param parameter the parameter whose dependency should be resolved (must not be * {@code null}) * @param parameterIndex the index of the parameter in the constructor or method * that declares the parameter * @param containingClass the concrete class that contains the parameter; this may * differ from the class that declares the parameter in that it may be a subclass * thereof, potentially substituting type variables (must not be {@code null}) * @param beanFactory the {@code AutowireCapableBeanFactory} from which to resolve * the dependency (must not be {@code null}) * @return the resolved object, or {@code null} if none found * @throws BeansException if dependency resolution failed * @see #isAutowirable * @see Autowired#required * @see SynthesizingMethodParameter#forExecutable(Executable, int) * @see AutowireCapableBeanFactory#resolveDependency(DependencyDescriptor, String) */ @Nullable public static Object resolveDependency(Parameter parameter, int parameterIndex, Class containingClass, AutowireCapableBeanFactory beanFactory) throws BeansException { Assert.notNull(parameter, "Parameter is required"); Assert.notNull(containingClass, "Containing class is required"); Assert.notNull(beanFactory, "AutowireCapableBeanFactory is required"); AnnotatedElement annotatedParameter = getEffectiveAnnotatedParameter(parameter, parameterIndex); Autowired autowired = AnnotatedElementUtils.findMergedAnnotation(annotatedParameter, Autowired.class); boolean required = (autowired == null || autowired.required()); MethodParameter methodParameter = SynthesizingMethodParameter.forExecutable( parameter.getDeclaringExecutable(), parameterIndex); DependencyDescriptor descriptor = new DependencyDescriptor(methodParameter, required); descriptor.setContainingClass(containingClass); return beanFactory.resolveDependency(descriptor, null); } /** * Due to a bug in {@code javac} on JDK versions prior to JDK 9, looking up * annotations directly on a {@link Parameter} will fail for inner class * constructors. *

Bug in javac in JDK < 9

*

The parameter annotations array in the compiled byte code excludes an entry * for the implicit enclosing instance parameter for an inner class * constructor. *

Workaround

*

This method provides a workaround for this off-by-one error by allowing the * caller to access annotations on the preceding {@link Parameter} object (i.e., * {@code index - 1}). If the supplied {@code index} is zero, this method returns * an empty {@code AnnotatedElement}. *

WARNING

*

The {@code AnnotatedElement} returned by this method should never be cast and * treated as a {@code Parameter} since the metadata (e.g., {@link Parameter#getName()}, * {@link Parameter#getType()}, etc.) will not match those for the declared parameter * at the given index in an inner class constructor. * * @return the supplied {@code parameter} or the effective {@code Parameter} * if the aforementioned bug is in effect */ private static AnnotatedElement getEffectiveAnnotatedParameter(Parameter parameter, int index) { Executable executable = parameter.getDeclaringExecutable(); if (executable instanceof Constructor && ClassUtils.isInnerClass(executable.getDeclaringClass()) && executable.getParameterAnnotations().length == executable.getParameterCount() - 1) { // Bug in javac in JDK <9: annotation array excludes enclosing instance parameter // for inner classes, so access it with the actual parameter index lowered by 1 return index == 0 ? EMPTY_ANNOTATED_ELEMENT : executable.getParameters()[index - 1]; } return parameter; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy