Spring进阶篇(1)-AutowireCapableBeanFactory(容器外的Bean使用依赖注入)

JAVA && Spring && SpringBoot2.x — 学习目录

如何实现不在IOC容器的Bean也可以被Spring管理呢?

平时我们的做法,在IOC容器里的类,实现一个ApplicationContextAware接口获取ApplicationContext上下文对象,通过getBean的方式,获取到Spring容器的Bean。如何实现ApplicationContextAware接口,可以参考IOC-如何动态获取spring容器中的bean对象(ApplicationContextAware)

但是这篇文章,重点说的是AutowireCapableBeanFactory接口。

AutowireCapableBeanFactory是在BeanFactory的基础上实现对已存在实例的管理。可以使用这个接口集成其他框架,捆绑并填充并不由Spring管理生命周期并已存在的实例。

需要注意的是,ApplicationContext接口没有实现AutowireCapableBeanFactory接口,因为应用代码很少用到此功能,如果需要的话,可以调用ApplicationContext的getAutowireCapableBeanFactory方法,来获取此接口的实例。

AutowireCapableBeanFactory定义了5种装配策略:

  1. 不自动注入:AUTOWIRE_NO
  2. 使用BeanName策略注入:AUTOWIRE_BY_NAME
  3. 使用类型装配策略:AUTOWIRE_BY_TYPE
  4. 使用构造器装配策略:AUTOWIRE_CONSTRUCTOR
  5. 自动装配策略:AUTOWIRE_AUTODETECT

一般我们采用下面的方法就可实现属性的依赖注入。

void autowireBean(Object existingBean) throws BeansException; // 使用autowireBeanProperties装配属性

该方法实现类是在AbstractAutowireCapableBeanFactory中。

public void autowireBean(Object existingBean) {
        // Use non-singleton bean definition, to avoid registering bean as dependent bean.
        RootBeanDefinition bd = new RootBeanDefinition(ClassUtils.getUserClass(existingBean));
        bd.setScope(BeanDefinition.SCOPE_PROTOTYPE);
        bd.allowCaching = ClassUtils.isCacheSafe(bd.getBeanClass(), getBeanClassLoader());
        BeanWrapper bw = new BeanWrapperImpl(existingBean);
        initBeanWrapper(bw);
        populateBean(bd.getBeanClass().getName(), bd, bw);
    }

实际上该方法的逻辑主要是在populateBean中。这个方法是Spring中一个重要的方法。用于装配Bean。主要是通过反射获取到我们new出来的对象的属性及注解,若是注解时AutowiredValueInject时,进行Bean组装。此方法执行完毕,我们new出来的方法就可以通过注解注入的bean进行操作了。

实战运用

1. 拦截器或者监听器使用IOC管理的bean。

实际上在SpringMVC框架中更加明显,web.xml配置的执行顺序:监听器>过滤器>servlet。监听器或者过滤器并没有被Spring容器所管理,那么他们如何使用依赖注入呢?我们知道,在拦截器中有一个Filter交由Spring管理-DelegatingFilterProxy可以实现Spring管理过滤器,但是有其他方式实现呢?

1. web.xml中配置监听器

     
    
        com.springmvc.common.listener.MyServletContextListener
    

    
        myFilter
        com.springmvc.common.filter.MyFilter
    
    
        myFilter
        /*
    

监听器解决方案:

public class MyServletContextListener implements ServletContextListener {


    @Autowired
    private RoleService roleService;

    Logger logger = LoggerFactory.getLogger(MyServletContextListener.class);

    public void contextInitialized(ServletContextEvent sce) {

        AutowireCapableBeanFactory autowireCapableBeanFactory = WebApplicationContextUtils
                .getWebApplicationContext(sce.getServletContext()).getAutowireCapableBeanFactory();
        autowireCapableBeanFactory.autowireBean(this);
        JSONObject sysRescTreeJson = roleService.getSysRescTreeJson(1);
        logger.info("【监听器打印:】" + JSON.toJSONString(sysRescTreeJson));
    }
    public void contextDestroyed(ServletContextEvent sce) {

    }
}

拦截器解决方案:

public class MyFilter implements Filter {

    private static Logger logger = LoggerFactory.getLogger(MyFilter.class);

    @Autowired
    private RoleService roleService;


    public void init(FilterConfig filterConfig) throws ServletException {
        ServletContext servletContext = filterConfig.getServletContext();
        AutowireCapableBeanFactory autowireCapableBeanFactory = WebApplicationContextUtils.getWebApplicationContext(servletContext).getAutowireCapableBeanFactory();
        autowireCapableBeanFactory.autowireBean(this);
    }

    public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
        logger.info("【过滤器 test】");
        JSONObject sysRescTreeJson = roleService.getSysRescTreeJson(1);
        chain.doFilter(request,response);

    }

    public void destroy() {

    }
}

Quartz框架的解决方案

Job的创建是有Quartz通过反射创建的,并未交由Spring容器创建。故原则上来说,是无法在Job实例中使用依赖注入的,解决方案如下:

Spring/Spring boot正确集成Quartz及解决@Autowired失效问题

实际上,在SpringBoot2.0以上的版本,若是我们引入了

 
        org.springframework.boot
        spring-boot-starter-quartz
    

org.springframework.boot.autoconfigure.quartz.QuartzAutoConfiguration类中,进行自动装配。

@Bean
    @ConditionalOnMissingBean
    public SchedulerFactoryBean quartzScheduler() {
        //解决创建的bean不由IOC管理却使用依赖注入的问题
        SchedulerFactoryBean schedulerFactoryBean = new SchedulerFactoryBean();
        SpringBeanJobFactory jobFactory = new SpringBeanJobFactory();
        jobFactory.setApplicationContext(this.applicationContext);
        schedulerFactoryBean.setJobFactory(jobFactory);
        //scheduler的配置代码,省略     

        return schedulerFactoryBean;
    }

关键代码

public class SpringBeanJobFactory extends AdaptableJobFactory
        implements ApplicationContextAware, SchedulerContextAware {
    @Nullable
    private ApplicationContext applicationContext;
    @Override
    public void setApplicationContext(ApplicationContext applicationContext) {
        this.applicationContext = applicationContext;
    }
    @Override
    protected Object createJobInstance(TriggerFiredBundle bundle) throws Exception {
        Object job = (this.applicationContext != null ?
      //使用AutowireCapableBeanFactory完成对IOC外Bean的操作
            this.applicationContext.getAutowireCapableBeanFactory().createBean(
                        bundle.getJobDetail().getJobClass(), AutowireCapableBeanFactory.AUTOWIRE_CONSTRUCTOR, false) :
                super.createJobInstance(bundle));
    //创建Job的代码,可忽略。
        return job;
    }

}

你可能感兴趣的:(Spring进阶篇(1)-AutowireCapableBeanFactory(容器外的Bean使用依赖注入))