繁体   English   中英

在实例化之前自定义Spring bean自动装配构造函数中的参数

[英]Customize arguments in spring bean autowired constructor, before instantiation

我需要实现的是与@RequestParam("foo")相同的行为,但是在一个REQUEST范围内的bean的构造函数中,其他构造函数参数将正常进行自动装配。

我想在构造函数中注入一个请求参数,就好像它是普通bean一样,但是显然没有实际是spring bean。

例:

@Component
@RequestScope
public class RequestBean {    
    private final String filter;
    private final UserRepository userRepository;

    public RequestBean(@Param("filter") String filter, UserRepository userRepository) {
        this.filter = filter;
        this.userRepository = userRepository;
    }
    ...
}

我尝试使用BeanFactoryPostProcessor,但必须在请求时而不是在bean定义时获取需要注入的值。

public class MyBeanFactoryPostProcessor implements BeanFactoryPostProcessor {

    @Override
    public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {

        Stream.of(beanFactory.getBeanDefinitionNames())
                .map(name -> beanFactory.getBeanDefinition(name))
                .filter(def -> "request".equals(def.getScope()))
                .forEach(beanDefinition -> {
                    try {
                        System.out.println(beanDefinition);

                        for (Constructor<?> constructor : Class.forName(beanDefinition.getBeanClassName()).getConstructors()) {
                            for (Parameter parameter : constructor.getParameters()) {
                                Param param = parameter.getAnnotation(Param.class);
                                if(param != null) {
                                    String paramName = param.value(); // filter
                                    Object requestParameterValue = "request.getParameter('" + paramName + "')"; // bean creation time evaluated argument

                                    beanDefinition.getConstructorArgumentValues().addGenericArgumentValue(
                                        new ConstructorArgumentValues.ValueHolder(requestParameterValue, parameter.getType().getName(), parameter.getName())
                                    );      
                                }
                            }
                        }
                    } catch (Exception ex) {
                        // whatever
                    }
                });
    }
}

对于对/index?filter=blue的请求,我希望“ blue”和userRepository已自动连线。

如果我尝试使用固定值,那么它可以完美工作,该值和另一个bean可以正常自动接线。

我无法使用BeanPostProcessor因为必须在实例化(构造函数注入)之前获取值

最后,我找到了实现自己想要的方法。

我定义了一个REQUEST范围,以简化获取和转换请求参数的过程。 由于在BeanFactoryPostProcessor中注册了一个表达式(SprEL),因此将在创建时调用此bean。 将对表达式进行求值,并将返回值注入到带注释的参数中。

这会将动态表达式添加到构造函数参数中:

String name = param.value();
TypedStringValue dynamicValue = new TypedStringValue("#{@myBean.getParameter('"+name+"')}");
beanDefinition.getConstructorArgumentValues().addIndexedArgumentValue(index, dynamicValue, constructorParameter.getType().getName());

这是在带注释的bean实例化之前由表达式调用的bean:

@Component
@RequestScope
class MyBean {
    final HttpServletRequest request;

    public MyBean(HttpServletRequest request) {
        this.request = request;
    }        

    public String getParameter(String name){
        // do something 
        return request.getParameter(name);
    }
}

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM