zoukankan      html  css  js  c++  java
  • Spring 中的 BeanFactory 与 FactoryBean

    1.前提概要

    很多java开发者在使用Spring框架中都见过后缀为FactoryBean的类,比如Mybatis-Spring中的SqlSessionFactoryBean。说到这里就不得不提BeanFactory。FactoryBean和BeanFactory特别容易让人混淆,面试还经常问到这两种概念。其实它们的作用和使用场景是不一样的

    2.BeanFactory

    先来说说BeanFactory。 用于访问Spring bean容器的根接口。这是Spring bean容器的基本客户端视图。原来是获取Spring Bean的接口,也就是IoC容器。然后我们看类图

    原来我们更常用的ApplicationContext就是一个BeanFactory。我们通过bean的名称或者类型都可以从BeanFactory来获取bean。对于BeanFactory这么介绍相信都不陌生了。让我们把关注点转向FactoryBean上。

    3.FactoryBean

    FactoryBean 是个什么玩意儿呢?来看看源码

              public interface FactoryBean<T> {
    
    	          @Nullable
    	         T getObject() throws Exception;
    
     
    	          @Nullable
    	         Class<?> getObjectType();
    
     
    	         default boolean isSingleton() {
    		       return true;
    	         }
             }
    
    • T getObject() 获取泛型T的实例。用来创建Bean。当IoC容器通过getBean方法来FactoryBean创建的实例时实际获取的不是FactoryBean 本身而是具体创建的T泛型实例。等下我们会来验证这个事情。
    • Class<?> getObjectType() 获取 T getObject()中的返回值 T 的具体类型。这里强烈建议如果T是一个接口,返回其具体实现类的类型。
    • default boolean isSingleton() 用来规定 Factory创建的的bean是否是单例。这里通过默认方法定义为单例。

    3.1 FactoryBean使用场景

    FactoryBean 用来创建一类bean。比如你有一些同属鸟类的bean需要被创建,但是它们自己有各自的特点,你只需要把他们的特点注入FactoryBean中就可以生产出各种鸟类的实例。举一个更加贴近实际生产的例子。甚至这个例子你可以应用到实际java开发中去。我们需要自己造一个定时任务的轮子。用FactoryBean 再合适不过了。我们来用代码说话一步步来演示FactoryBean的使用场景。

    3.2 构建一个FactoryBean

    我们声明定时任务一般具有下列要素:

    • 时间周期,肯定会使用到cron表达式。
    • 一个任务的执行抽象接口。
    • 定时任务具体行为的执行者。

    Task任务执行抽象接口的实现。实现包含两个方面:

    • SomeService 是具体任务的执行逻辑。
    • cron时间表达式
    
    public class CustomTask implements Task {
        private SomeService someService;
        private String cronExpression;
    
        public CustomTask(SomeService someService) {
            this.someService = someService;
        }
    
        @Override
        public void execute() {
            //do something
            someService.doTask();
        }
    
        @Override
        public void setCron(String cronExpression) {
            this.cronExpression = cronExpression;
        }
    
        @Override
        public String getCron() {
            return cronExpression;
        }
    }
    
    

    通过以上的定义。任务的时间和任务的逻辑可以根据不同的业务做到差异化配置。然后我们实现一个关于Task的FactoryBean。

    
    public class TaskFactoryBean implements FactoryBean<Task> {
        private SomeService someService;
        private String cronExpression;
    
    
        @Override
        public Task getObject() throws Exception {
            CustomTask customTask = new CustomTask(someService);
            customTask.setCron(cronExpression);
            return customTask;
        }
    
        @Override
        public Class<?> getObjectType() {
            return CustomTask.class;
        }
    
        @Override
        public boolean isSingleton() {
            return true;
        }
    
        public SomeService getSomeService() {
            return someService;
        }
    
        public void setSomeService(SomeService someService) {
            this.someService = someService;
        }
    
        public String getCronExpression() {
            return cronExpression;
        }
    
        public void setCronExpression(String cronExpression) {
            this.cronExpression = cronExpression;
        }
    }
    
    
    

    3.3 FactoryBean 注入IoC

    你可以使用xml的注入方式,当然也可以使用javaConfig的配置方式。这里我们使用javaConfig注入。我们将两个FactroyBean注入到Spring容器中去。

    @Configuration
    public class Config {
    
        @Bean
        public TaskFactoryBean customTask() {
            TaskFactoryBean taskFactoryBean = new TaskFactoryBean();
            taskFactoryBean.setCronExpression("0 15 10 * * ?");
            String word = "定时任务一";
            SomeService someService = new SomeService();
            someService.setWord(word);
            taskFactoryBean.setSomeService(someService);
            return taskFactoryBean;
        }
    
        @Bean
        public TaskFactoryBean otherTask() {
            TaskFactoryBean taskFactoryBean = new TaskFactoryBean();
            taskFactoryBean.setCronExpression("0 15 17 * * ?");
            String word = "定时任务二";
            SomeService someService = new SomeService();
            someService.setWord(word);
            taskFactoryBean.setSomeService(someService);
            return taskFactoryBean;
        }
    }
    

    3.4 FactoryBean的一些特点

    一般如上声明后,@Bean注解如果不显式声明bean名称则方法名作为bean的名称,而且返回值作为注入的Bean。但是我们通过debug发现却是这样的:

    也就是说通过方法名是返回FactoryBean 创建的Bean。那么如何返回该FactoryBean呢?上图中也给出了答案在方法前增加引用符“&”。具体的原因还用从BeanFactory中寻找,真是不是冤家不聚头

    我们对上面声明的两个bean进行测试,也出色地完成了不同的定时任务业务逻辑。

        @Autowired
        private Task customTask;
        @Autowired
        private Task otherTask;
    
    
        @Test
        public void task() {
            customTask.execute();
            otherTask.execute();
        }
    

    4. 总结

    在后续的使用中你可以通过声明不同的cron表达式,以及不同SomeService来定制更多的定时任务。通过这个例子相信你会对FactoryBean有的清晰的认识。demo就不提供了,非常简单,强烈建议你自己试一试以加深理解。

    关注微信公众号:码农小胖哥 获取更多干货

  • 相关阅读:
    org.hibernate.HibernateException: No Session found for current thread
    TCP/IP协议 HTTP协议
    [ERROR][org.springframework.web.context.ContextLoader][main] Context initialization failed org.sprin
    oracle 导出表结构和数据,使用plsql
    jar包里查找指定的class文件,排查是否存在或重复,工具软件:Java Class Finder
    maven手动安装jar到本地仓库
    The reference to entity “idNo” must end with the ';' delimiter 异常处理
    activeMQ下载,安装,启动,关闭
    Oracle错误:ORA-01033
    -Xms512m -Xmx1024m -XX:PermSize=256m -XX:MaxPermSize=512m
  • 原文地址:https://www.cnblogs.com/felordcn/p/12142572.html
Copyright © 2011-2022 走看看