Spring中BeanFactory与FactoryBean接口的区别详解

前言

Spring框架中的BeanFactory接口和FactoryBean接口因为名称相似,老是容易搞混淆,而且也是面试过程中经常会碰到的一个问题。所以本文就专门给大家整理出来。

一、BeanFactory接口

BeanFactory接口是Spring容器的核心接口,负责:实例化、定位、配置应用程序中的对象及建立这些对象间的依赖。

Spring为我们提供了许多易用的BeanFactory实现,XmlBeanFactory就是常用的一个,该实现将以XML方式描述组成应用的对象及对象间的依赖关系。XmlBeanFactory类将持有此XML配置元数据,并用它来构建一个完全可配置的系统或应用。

BeanFactory bf = new XmlBeanFactory(new ClassPathResource("applicationContext.xml"));
Object bean = bf.getBean(IUserService.class);
System.out.println(bean);

接口中定义的方法

public interface BeanFactory {

 String FACTORY_BEAN_PREFIX = "&";

 Object getBean(String name) throws BeansException;

 <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;

 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;

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

 String[] getAliases(String name);
}

二、FactoryBean接口

BeanFactory接口是Spring的核心接口。功能非常复杂,这个时候如果我们想要编写一些比较复杂点儿的逻辑就会触及到其他一些不必要的接口,不好实现。这时候使用FactoryBean就比较方便了。FactoryBean以Bean结尾是个Bean对象,不是工厂。接口中定义的方法如下:

public interface FactoryBean<T> {

 /**
  * 返回对象的实例
  */
 T getObject() throws Exception;

 /**
  * 返回对象的类型
  */
 Class<?> getObjectType();

 /**
  * 是否是单例
  */
 boolean isSingleton();
}

1.简单实现

接口和实现类

public interface IUserService {

 public void doSome();
}
public class UserServiceImpl implements IUserService {
 public UserServiceImpl(){
  System.out.println("--被实例化了--");
 }

 @Override
 public void doSome() {
  System.out.println("UserServiceImpl 。。。 被执行了");
 }
}

自定义FactoryBean

public class MyFactoryBean implements FactoryBean<IUserService>{

 @Override
 public IUserService getObject() throws Exception {
  System.out.println("--IUserService实例化之前---");
  IUserService service = new UserServiceImpl();
  System.out.println("--IUserService实例化后---");
  return service;
 }

 @Override
 public Class<?> getObjectType() {
  return IUserService.class;
 }

 @Override
 public boolean isSingleton() {
  return true;
 }
}

配置文件

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
 xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
 xsi:schemaLocation="http://www.springframework.org/schema/beans
 http://www.springframework.org/schema/beans/spring-beans.xsd">
 <bean id="myFactoryBean" class="com.dpb.factorybean.MyFactoryBean"/>
</beans>

测试--通过类型获取

@Test
public void test1() {
 BeanFactory bf = new XmlBeanFactory(new ClassPathResource("applicationContext.xml"));
 Object bean = bf.getBean(IUserService.class);
 System.out.println(bean);
}

输出结果

--IUserService实例化之前---
--被实例化了--
--IUserService实例化后---
com.dpb.service.UserServiceImpl@5315b42e

测试--通过id获取

@Test
public void test1() {
 BeanFactory bf = new XmlBeanFactory(new ClassPathResource("applicationContext.xml"));
 Object bean = bf.getBean("myFactoryBean");
 System.out.println(bean);
}

输出结果

--IUserService实例化之前---
--被实例化了--
--IUserService实例化后---
com.dpb.service.UserServiceImpl@783e6358

如果想要获取FactoryBean对象 id前加 &就可以

@Test
public void test1() {
 BeanFactory bf = new XmlBeanFactory(new ClassPathResource("applicationContext.xml"));
 Object bean = bf.getBean("&myFactoryBean");
 System.out.println(bean);
}

输出结果

com.dpb.factorybean.MyFactoryBean@3b81a1bc

2.增强实现

通过FactoryBean创建一个代理类来增强目标类,我们来看下效果

接口和实现类

public interface IUserService {

 public void doSome();
}
public class UserServiceImpl implements IUserService {
 public UserServiceImpl(){
  System.out.println("--被实例化了--");
 }

 @Override
 public void doSome() {
  System.out.println("UserServiceImpl 。。。 被执行了");
 }
}

自定义FactoryBean

public class MyFactoryBean implements FactoryBean,InitializingBean,DisposableBean{

 private Object proxyObject;

 private Object target;

 private String interfaceName;

 @Override
 public Object getObject() throws Exception {

  return proxyObject;
 }

 @Override
 public Class<?> getObjectType() {
  return proxyObject.getClass()==null?Object.class:proxyObject.getClass();
 }

 @Override
 public boolean isSingleton() {
  return true;
 }

 /**
  * MyFactoryBean 对象销毁的回调方法
  * @throws Exception
  */
 @Override
 public void destroy() throws Exception {
  System.out.println("destroy ....");

 }

 /**
  * MyFactoryBean 对象实例化的方法
  */
 @Override
 public void afterPropertiesSet() throws Exception {
  System.out.println("---afterPropertiesSet---");
  proxyObject = Proxy.newProxyInstance(
     this.getClass().getClassLoader()
     , new Class[]{Class.forName(interfaceName)}
     , new InvocationHandler() {

      @Override
      public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
       System.out.println("----代理方法执行开始----");
       Object obj = method.invoke(target, args);
       System.out.println("----代理方法执行结束----");
       return obj;
      }
     });
 }

 public String getInterfaceName() {
  return interfaceName;
 }

 public void setInterfaceName(String interfaceName) {
  this.interfaceName = interfaceName;
 }

 public Object getTarget() {
  return target;
 }

 public void setTarget(Object target) {
  this.target = target;
 }
}

配置文件

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
 xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
 xsi:schemaLocation="http://www.springframework.org/schema/beans
 http://www.springframework.org/schema/beans/spring-beans.xsd">
 <!-- 注册目标对象 -->
 <bean class="com.dpb.service.UserServiceImpl" id="userServiceImpl"/>
 <!-- 注册FactoryBean对象 -->
 <bean id="myFactoryBean" class="com.dpb.factorybean.MyFactoryBean">
  <property name="interfaceName" value="com.dpb.service.IUserService"/>
   <property name="target" ref="userServiceImpl"/>
 </bean>
</beans>

测试

@Test
public void test1() {
 ApplicationContext ac = new ClassPathXmlApplicationContext("applicationContext.xml");
 IUserService bean = (IUserService) ac.getBean("myFactoryBean");
 System.out.println("****************");
 bean.doSome();
 System.out.println();
}

输出结果:

--被实例化了--
---afterPropertiesSet---
****************
----代理方法执行开始----
UserServiceImpl 。。。 被执行了
----代理方法执行结束----

小结:通过输出结果我们可以看到通过FactoryBean接口我们也可以实现BeanFactory中某些接口提供的功能,而且会更加的灵活一些。

3.FactoryBean的实际使用案例

Spring在整合mybatis框架的时候提供的SqlSessionFactoryBean就是FactoryBean的很好的实现。

<!-- 整合mybatis -->
<bean class="org.mybatis.spring.SqlSessionFactoryBean"
id="sqlSessionFactoryBean" >
 <!-- 关联数据源 -->
 <property name="dataSource" ref="dataSource"/>
 <!-- 关联mybatis的配置文件 -->
 <property name="configLocation" value="classpath:mybatis-cfg.xml"/>
 <!-- 添加别名设置 -->
 <property name="typeAliasesPackage" value="com.sxt.model"/>
 <!-- 映射文件和接口文件不在同一个目录下的时候,需要单独指定映射文件的路径 -->
 <property name="mapperLocations" value="classpath:mapper/*.xml"/>
</bean>

Spring会调用SqlSessionFactoryBean这个实现了FactoryBean的工厂Bean 同时加载dataSource,Mapper文件的路径,对sqlSessionFactory进行初始化。

源代码比较多就不一一贴出来。到了这儿可以自行跟踪下源代码。

核心方法

@Override
 public void afterPropertiesSet() throws Exception {
 // 省略
 this.sqlSessionFactory = buildSqlSessionFactory();
 }

 protected SqlSessionFactory buildSqlSessionFactory() throws IOException {

 Configuration configuration;

 XMLConfigBuilder xmlConfigBuilder = null;
 // 省略
 return this.sqlSessionFactoryBuilder.build(configuration);
 }
/**
 * {@inheritDoc}
 */
 @Override
 public SqlSessionFactory getObject() throws Exception {
 if (this.sqlSessionFactory == null) {
  afterPropertiesSet();
 }

 return this.sqlSessionFactory;
 }

 /**
 * {@inheritDoc}
 */
 @Override
 public Class<? extends SqlSessionFactory> getObjectType() {
 return this.sqlSessionFactory == null ? SqlSessionFactory.class : this.sqlSessionFactory.getClass();
 }

 /**
 * {@inheritDoc}
 */
 @Override
 public boolean isSingleton() {
 return true;
 }

maven坐标:

<dependency>
 <groupId>org.mybatis</groupId>
 <artifactId>mybatis-spring</artifactId>
 <version>1.3.2</version>
</dependency>

三、总结

  • BeanFactory是Spring中IOC容器最核心的接口,遵循了IOC容器中所需的基本接口。例如我们很常见的:ApplicationContext,XmlBeanFactory 等等都使用了BeanFactory这个接口。
  • FactoryBean是工厂类接口,当你只是想简单的去构造Bean,不希望实现原有大量的方法。它是一个Bean,不过这个Bean能够做为工厂去创建Bean,同时还能修饰对象的生成。
  • FactoryBean比BeanFactory在生产Bean的时候灵活,还能修饰对象,带有工厂模式和装饰模式的意思在里面,不过它的存在还是以Bean的形式存在。

好了,以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对我们的支持。

(0)

相关推荐

  • 深入了解Spring中的FactoryBean

    FactoryBean和BeanFactory由于在命名上极其相似,一直以来困扰了不少的开发者. BeanFactory,耳熟能详的Spring核心接口,提供IoC容器的最基本功能.但要解释FactoryBean一句话可能就说不清楚了.我们将从下面的例子逐步说明,FactoryBean是什么,它提供了什么样的能力. /** * 布料 * 包含颜色属性 * Created by OKevin On 2019/9/3 **/ public class Cloth { private Red red;

  • 深入浅出重构Mybatis与Spring集成的SqlSessionFactoryBean(上)

    一般来说,修改框架的源代码是极其有风险的,除非万不得已,否则不要去修改.但是今天却小心翼翼的重构了Mybatis官方提供的与Spring集成的SqlSessionFactoryBean类,一来是抱着试错的心态,二来也的确是有现实需要. 先说明两点: 通常来讲,重构是指不改变功能的情况下优化代码,但本文所说的重构也包括了添加功能 本文使用的主要jar包(版本):spring-*-4.3.3.RELEASE.jar.mybatis-3.4.1.jar.mybatis-spring-1.3.0.jar

  • spring中的FactoryBean代码示例

    上篇文章中我们介绍了浅谈Spring的两种配置容器,接下来我们就了解下spring中的FactoryBean的相关内容,具体如下. 从SessionFactory说起: 在使用SSH集成开发的时候,我们有时候会在applicationContext.xml中配置Hibernate的信息,下面是配置SessionFactory的一段示例代码: <bean id="sessionFactory" class="org.springframework.orm.hibernat

  • spring中FactoryBean中的getObject()方法实例解析

    本文研究的主要是spring中FactoryBean中的getObject()方法的相关内容,具体如下. FactoryBean接口定义了以下3个接口方法: Object getObject():返回有FactoryBean创建的Bean实例,如果isSingleton()返回true,则该实例会放到Spring容器的单实例缓存池中. boolean isSingleton():确定由FactoryBean创建Bean的作用域是singleton还是prototype. Class getObj

  • spring中的BeanFactory与FactoryBean的讲解

    1.BeanFactory 1.1Spring提供了IOC容器的两种实现方式 ① BeanFactory:IOC容器的基本实现,是Spring内部的基础设施,是面向Spring本身的,不是提供给开发人员使用的. ② ApplicationContext:BeanFactory的子接口,提供了更多高级特性.面向Spring的使用者,几乎所有场合都使用ApplicationContext而不是底层的BeanFactory. 1.2 ApplicationContext的主要实现类 ClassPath

  • 详解Spring中的FactoryBean

    spring  FactoryBean 是创建 复杂的bean,一般的bean 直接用xml配置即可,如果一个bean的创建过程中涉及到很多其他的bean 和复杂的逻辑,用xml配置比较困难,这时可以考虑用FactoryBean 例子如下: 1:创建一个Car类(是为了简便)一般不能直接给出Car类,如果是这样直接注入就可以或者Car对象了,这里只是为了简便. package com.myapp.core.factorybean; public class Car { private Strin

  • Spring中BeanFactory与FactoryBean接口的区别详解

    前言 Spring框架中的BeanFactory接口和FactoryBean接口因为名称相似,老是容易搞混淆,而且也是面试过程中经常会碰到的一个问题.所以本文就专门给大家整理出来. 一.BeanFactory接口 BeanFactory接口是Spring容器的核心接口,负责:实例化.定位.配置应用程序中的对象及建立这些对象间的依赖. Spring为我们提供了许多易用的BeanFactory实现,XmlBeanFactory就是常用的一个,该实现将以XML方式描述组成应用的对象及对象间的依赖关系.

  • 简单了解Spring中BeanFactory与FactoryBean的区别

    这篇文章主要介绍了简单了解Spring中BeanFactory与FactoryBean的区别,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 在Spring中有BeanFactory和FactoryBean这2个接口,从名字来看很相似,比较容易搞混. 一.BeanFactory BeanFactory是一个接口,它是Spring中工厂的顶层规范,是SpringIoc容器的核心接口,它定义了getBean().containsBean()等管理Be

  • Spring中Bean的三种实例化方式详解

    目录 一.环境准备 二.构造方法实例化 三.分析Spring的错误信息 四.静态工厂实例化 4.1 工厂方式创建bean 4.2 静态工厂实例化 五.实例工厂与FactoryBean 5.1 环境准备 5.2 实例工厂实例化 5.3 FactoryBean的使用 六.bean实例化小结 一.环境准备 准备开发环境 创建一个Maven项目 pom.xml添加依赖 resources下添加spring的配置文件applicationContext.xml 最终项目的结构如下: 二.构造方法实例化 在

  • Java中Validated、Valid 、Validator区别详解

    目录 1. 结论先出 JSR 380 Valid VS Validated 不同点? Validator 2. @Valid和​​​​​​​@Validated 注解 3. 例子 4.使用@Valid嵌套校验 5. 组合使用@Valid和@Validated 进行集合校验 6. 自定义校验 自定义约束注解 工作原理 结论 参考链接: 1. 结论先出 Valid VS Validated 相同点 都可以对方法和参数进行校验 @Valid和@Validated 两种注释都会导致应用标准Bean验证.

  • Flutter 语法进阶抽象类和接口本质区别详解

    目录 1. 接口存在的意义? 2. 继承 VS 实现 3. Dart 中接口与实现的特殊性 4.Dart 中抽象类作为接口的小细节 1. 接口存在的意义? 在 Dart 中 接口 定义并没有对应的关键字.可能有些人觉得 Dart 中弱化了 接口 的概念,其实不然.我们一般对接口的理解是:接口是更高级别的抽象,接口中的方法都是 抽象方法 ,没有方法体.通过接口的定义,我们可以通过定义接口来声明功能,通过实现接口来确保某类拥有这些功能. 不过你有没有仔细想过,为什么接口会存在,引入接口的概念是为了解

  • C++ 中指针和引用有什么区别详解

    C++ 中指针和引用有什么区别详解 1.从内存上来讲  系统为指针分寸内存空间,而引用与绑定的对象共享内存空间,系统不为引用变量分配内容空间. 2指针初始化以后可以改变指向的对象,而引用定义的时候必须要初始化,且初始化以后不允许再重新绑定对象. 3.所以引用访问对象是直接访问.指针访问对象是间接访问. 4.如果pa是指针,那么*pa就是引用了. 但是两者在作为形参的时候非常相似,区别是指针拷贝副本,引用不拷贝.程序如下: #include<stdio.h> void pt(int * pta,

  • Python中set与frozenset方法和区别详解

    set(可变集合)与frozenset(不可变集合)的区别: set无序排序且不重复,是可变的,有add(),remove()等方法.既然是可变的,所以它不存在哈希值.基本功能包括关系测试和消除重复元素. 集合对象还支持union(联合), intersection(交集), difference(差集)和sysmmetric difference(对称差集)等数学运算. sets 支持 x in set, len(set),和 for x in set.作为一个无序的集合,sets不记录元素位

  • 基于js中style.width与offsetWidth的区别(详解)

    作为一个初学者,经常会遇到在获取某一元素的宽度(高度.top值...)时,到底是用 style.width还是offsetWidth的疑惑. 1. 当样式写在行内的时候,如 <div id="box" style="width:100px">时,用 style.width或者offsetWidth都可以获取元素的宽度. 但是,当样式写在样式表中时,如 #box{ width: 100px; }, 此时只能用offsetWidth来获取元素的宽度,而sty

  • 对python 中re.sub,replace(),strip()的区别详解

    1.strip(): str.strip([chars]);去除字符串前面和后面的所有设置的字符串,默认为空格 chars -- 移除字符串头尾指定的字符序列. st = " hello " st = st.strip() print(st+"end") 输出: 如果设置了字符序列的话,那么它会删除,字符串前后出现的所有序列中有的字符.但不会清除空格. st = "hello" st = st.strip('h,o,e') print(st) 因

  • 对python中数组的del,remove,pop区别详解

    以a=[1,2,3] 为例,似乎使用del, remove, pop一个元素2 之后 a都是为 [1,3], 如下: >>> a=[1,2,3] >>> a.remove(2) >>> a [1, 3] >>> a=[1,2,3] >>> del a[1] >>> a [1, 3] >>> a= [1,2,3] >>> a.pop(1) 2 >>>

随机推荐