在实例化之前,在spring bean autowired构造函数中自定义参数

时间:2016-11-21 17:30:04

标签: java spring spring-mvc

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

我想在构造函数中注入一个请求参数,就像它是一个普通的bean一样,但显然没有真正的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的请求,我希望&#34; blue&#34;和userRepository已自动装配。

如果我尝试使用固定值,它可以正常工作,值和其他bean正常自动装配。

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

1 个答案:

答案 0 :(得分:1)

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

我定义了一个String name = param.value(); TypedStringValue dynamicValue = new TypedStringValue("#{@myBean.getParameter('"+name+"')}"); beanDefinition.getConstructorArgumentValues().addIndexedArgumentValue(index, dynamicValue, constructorParameter.getType().getName()); 作用域,以简化获取和转换请求参数的过程。 由于在BeanFactoryPostProcessor中注册了表达式(SprEL),因此将在创建时调用此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);
    }
}

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

header=None