搞定Spring的BeanFactory和FactoryBean

Spring的核心是IOC和AOP,面试Java的基本上都会被问到它们是干什么的,可能你就简单的说了是控制反转与面向切面编程,那么具体细节就很可能被问懵。今天我们来讲一下IOC中围绕Bean的一个知识点,BeanFactory与FactoryBean。每天搞懂一个知识点,长期下来会让你受益匪浅。

一、区别

  • 相同点:

    BeanFactory是接口,提供了IOC容器最基本的形式,也就是SpringIOC所遵循的最底层和最基本的编程规范。

    FactoryBean是接口,实现这个接口的类,在注册到BeanFactory中时,并不像其他类暴露的是自己,而是FactoryBean中的getObject()方法的返回对象。

  • 不同点:

    BeanFactory是个factory,也就是IOC容器或者对相关工厂,在Spring中,所有的Bean都是由BeanFactory,也就是IOC容器来进行管理的。

    FactoryBean是个Bean,但是这个Bean不是一个简单的Bean,而是一个能生产或者修饰对象的工厂Bean,它的实现类似于设计模式中的工厂模式或者装饰者模式。

二、BeanFactory

比较熟悉的DefaultListableBeanFactory、XmlBeanFactory、ApplicationContext类都是BeanFactory的子类。

BeanFacotry是spring中比较原始的Factory。如XMLBeanFactory就是一种典型的BeanFactory。

原始的BeanFactory无法支持spring的许多插件,如AOP功能、Web应用等。ApplicationContext接口,它由BeanFactory接口派生而来,ApplicationContext包含BeanFactory的所有功能,通常建议比BeanFactory优先 。

ApplicationContext以一种更向面向框架的方式工作以及对上下文进行分层和实现继承,ApplicationContext包还提供了以下的功能:

  • MessageSource, 提供国际化的消息访问;
  • 资源访问,如URL和文件;
  • 事件传播;
  • 载入多个(有继承关系)上下文 ,使得每一个上下文都专注于一个特定的层次,比如应用的web层;

BeanFactory源码:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39

public interface BeanFactory {

String FACTORY_BEAN_PREFIX = "&";

Object getBean(String name) throws BeansException;

// 根据名称和类型获取,如果名称对应的bean不是requiredType类型的,就抛出错误
<T> T getBean(String name, Class<T> requiredType) throws BeansException;

Object getBean(String name, Object... args) throws BeansException;

<T> T getBean(Class<T> requiredType) throws BeansException;

<T> T getBean(Class<T> requiredType, Object... args) throws BeansException;

<T> ObjectProvider<T> getBeanProvider(Class<T> requiredType);

<T> ObjectProvider<T> getBeanProvider(ResolvableType requiredType);

boolean containsBean(String name);

boolean isSingleton(String name) throws NoSuchBeanDefinitionException;

boolean isPrototype(String name) throws NoSuchBeanDefinitionException;

boolean isTypeMatch(String name, ResolvableType typeToMatch) throws NoSuchBeanDefinitionException;

boolean isTypeMatch(String name, Class<?> typeToMatch) throws NoSuchBeanDefinitionException;

@Nullable
Class<?> getType(String name) throws NoSuchBeanDefinitionException;

@Nullable
Class<?> getType(String name, boolean allowFactoryBeanInit) throws NoSuchBeanDefinitionException;

String[] getAliases(String name);

}

方法简介:

  • boolean containsBean(String beanName)

    判断工厂中是否包含给定名称的bean定义,若有则返回true

  • Object getBean(String)

    返回给定名称注册的bean实例。根据bean的配置情况,如果是singleton模式将返回一个共享实例,否则将返回一个新建的实例,如果没有找到指定bean,该方法可能会抛出异常

  • Object getBean(String, Class)

    返回以给定名称给定类型注册的bean实例,如果bean不是给定Class的类型就抛出错误

  • Class getType(String name)

    返回给定名称的bean的Class,如果没有找到指定的bean实例,则排除NoSuchBeanDefinitionException异常

  • boolean isSingleton(String)

    判断给定名称的bean定义是否为单例模式

  • String[] getAliases(String name)

    返回给定bean名称的所有别名

    三、FactoryBean

    一般情况下,Spring通过反射机制利用<bean>的class属性指定实现类实例化Bean,在某些情况下,实例化Bean过程比较复杂,如果按照传统的方式,则需要在<bean>中提供大量的配置信息。配置方式的灵活性是受限的,这时采用编码的方式可能会得到一个简单的方案。Spring为此提供了一个org.springframework.bean.factory.FactoryBean的工厂类接口,用户可以通过实现该接口定制实例化Bean的逻辑。FactoryBean接口对于Spring框架来说占用重要的地位,Spring自身就提供了70多个FactoryBean的实现。它们隐藏了实例化一些复杂Bean的细节,给上层应用带来了便利。从Spring3.0开始,FactoryBean开始支持泛型,即接口声明改为FactoryBean<T>的形式

以Bean结尾,表示它是一个Bean,不同于普通Bean的是:它是实现了FactoryBean接口的Bean,根据该Bean的ID从BeanFactory中获取的实际上是FactoryBean的getObject()返回的对象,而不是FactoryBean本身,如果要获取FactoryBean对象,请在id前面加一个&符号来获取。

查看评论