Spring中FactoryBean的高级用法实战

news/2024/12/23 6:39:30/
❃博主首页 : 「码到三十五」 ,同名公众号 :「码到三十五」,wx号 : 「liwu0213」
☠博主专栏 : <mysql高手> <elasticsearch高手> <源码解读> <java核心> <面试攻关>
♝博主的话 : 搬的每块砖,皆为峰峦之基;公众号搜索「码到三十五」关注这个爱发技术干货的coder,一起筑基

文章目录

      • 前言
      • 基本用法
      • 创建多例对象
      • 创建代理
      • 获取原始 FactoryBean
      • 利用 SPI 获取对象
      • ServiceListFactoryBean获取所有SPI对象
      • FactoryBean应用场景
          • 1. 延迟初始化
          • 2. 依赖注入的高级用法
          • 3. 集成第三方库
          • 4. 自定义作用域
          • 5. 工厂方法的封装
          • 6. 与AOP集成
          • 7. 复杂依赖的解耦

前言

FactoryBean 是 Spring 框架中的一个高级特性,它允许开发者通过自定义的方式控制对象的创建过程。当需要编写复杂的初始化逻辑,而这些逻辑不适合直接放在类的构造函数或初始化方法中时,FactoryBean 提供了一个很好的解决方案。通过实现 FactoryBean 接口,可以在一个单独的方法(通常是 getObject())中封装所有的初始化逻辑,并将这个逻辑的结果(即对象实例)返回给 Spring 容器。

FactoryBean 接口定义了三个关键方法:

  • T getObject(): 返回由该工厂创建的对象的实例。
  • boolean isSingleton(): 指定返回的实例是否为单例。
  • Class<?> getObjectType(): 返回 getObject() 方法返回的对象类型。

Spring 框架内部广泛使用 FactoryBean,提供了超过50个 FactoryBean 的实现,用于创建和配置各种复杂的对象。

下面基于 UserService 类,通过几个实战案例来展示 FactoryBean 的不同用法。


class UserService {public void save() {System.out.println("save user ...") ;}
}

基本用法

在这个例子中,我们创建了一个简单的 UserServiceFactoryBean,用于创建 UserService 的实例。

@Component
public class UserServiceFactoryBean implements FactoryBean<UserService> {@Overridepublic UserService getObject() throws Exception {return new UserService();}@Overridepublic Class<?> getObjectType() {return UserService.class;}@Overridepublic boolean isSingleton() {return true; // 默认返回单例}
}

使用方式:

@Resource
private UserService userService; // 直接注入// 或者通过 ApplicationContext 获取
ApplicationContext context = ...;
UserService us = context.getBean(UserService.class);

创建多例对象

isSingleton() 方法返回 false,以创建多例对象。

@Override
public boolean isSingleton() {return false; // 返回非单例
}

此时,每次通过 Spring 容器获取的 UserService 实例都将是新的,以下两个Controller中注入的UserService将是两个不同的对象:


@Component
public class UserController {@Resourceprivate UserService userService ;
}
@Component
public class CommonService {@Resourceprivate UserService userService ;
}

创建代理

使用 FactoryBean 创建代理对象是一种常见用法,尤其是在需要为对象添加横切关注点(如日志、事务管理等)时。

@Override
public PersonService getObject() throws Exception {ProxyFactory factory = new ProxyFactory(new UserService());factory.addAdvice(new MethodInterceptor() {@Overridepublic Object invoke(MethodInvocation invocation) throws Throwable {System.out.println("Before method call");Object result = invocation.proceed();System.out.println("After method call");return result;}});return (UserService) factory.getProxy();
}

获取原始 FactoryBean

有时候,可能需要获取 FactoryBean 本身而不是它创建的对象。

// 通过类型获取
UserServiceFactoryBean factoryBean = context.getBean(UserServiceFactoryBean.class);// 或者通过名称获取,添加 '&' 前缀
UserServiceFactoryBean factoryBeanByName = context.getBean("&userServiceFactoryBean");

利用 SPI 获取对象

Spring 提供的 ServiceFactoryBeanServiceListFactoryBean 可以方便地用于基于 SPI 机制的服务加载。

@Configuration
public class AppConfig {@Beanpublic ServiceFactoryBean<DAO> daoFactoryBean() {ServiceFactoryBean<DAO> fb = new ServiceFactoryBean<>();fb.setServiceType(Partition.class);return fb;}
}

META-INF/services 目录下创建与 Partition 接口全限定名对应的文件,列出所有实现类的全限定名。Spring 将加载并实例化这些实现类,并通过 daoFactoryBean 提供访问。


com.diguobobo.helper.IdPartition
com.diguobobo.helper.DatePratition

容器中注入Partition时,将得到这里的第一个IdPartition实例。

ServiceListFactoryBean获取所有SPI对象

如果你需要获取 SPI 接口的所有实现,而不是单个实现,可以使用 ServiceListFactoryBean。这个 Bean 工厂会返回一个包含所有 SPI 实现的列表。配置方式与 ServiceFactoryBean 类似,但返回的将是一个列表,而不是单个对象。

@Configuration  
public class AppConfig {  @Bean  public ServiceListFactoryBean<DAO> daoListFactoryBean() {  ServiceListFactoryBean<DAO> fb = new ServiceListFactoryBean<>();  fb.setServiceType(Partition.class);  return fb;  }  
}  // 使用时注入List<Partition>  
@Autowired  
private List<Partition> daos;

FactoryBean应用场景

1. 延迟初始化

默认情况下,Spring容器中的Bean会在容器启动时进行初始化。但是,通过FactoryBean,你可以控制对象的创建时机,直到真正需要该对象时才进行创建。这可以通过在FactoryBean中实现特定的逻辑来延迟调用getObject()方法实现。

2. 依赖注入的高级用法

FactoryBean允许开发者在依赖注入过程中进行更精细的控制。例如,你可以根据特定的条件动态地选择不同的Bean实例进行注入,或者根据环境变量、配置属性等动态地创建Bean实例。

3. 集成第三方库

当需要将第三方库中的对象集成到Spring容器中时,如果这些对象的创建过程比较复杂或者不符合Spring的默认Bean创建规则,你可以通过实现FactoryBean来封装这些复杂的创建逻辑。这样,就可以像使用其他Spring Bean一样使用这些第三方库中的对象了。

4. 自定义作用域

虽然FactoryBean本身并不直接提供作用域的定义(作用域通常由Spring容器管理),但你可以通过FactoryBean来控制对象的创建过程,从而实现自定义作用域的效果。例如,可以通过FactoryBean来管理具有自定义生命周期的对象,如数据库连接、网络会话等。

5. 工厂方法的封装

有时可能需要使用某个类的静态工厂方法来创建对象实例。虽然Spring支持通过@Bean注解来引用静态工厂方法,但实现FactoryBean提供了一种更加封装和灵活的方式来处理这种情况。你可以在FactoryBean中实现调用静态工厂方法的逻辑,并将FactoryBean本身注册为Spring容器中的Bean。

6. 与AOP集成

FactoryBean可以与Spring的AOP(面向切面编程)功能集成,用于在对象创建过程中应用横切关注点(如事务管理、日志记录等)。虽然通常这些横切关注点会应用在Bean的方法调用上,但通过在FactoryBean中实现特定的逻辑,你也可以在对象创建过程中应用这些关注点。

7. 复杂依赖的解耦

复杂的应用中,Bean之间可能存在复杂的依赖关系。通过实现FactoryBean,你可以将这些复杂的依赖关系封装在FactoryBean内部,从而简化Bean之间的依赖关系。这样,其他Bean只需要依赖于FactoryBean创建的实例,而不需要关心这些实例背后的复杂创建逻辑和依赖关系。


关注公众号[码到三十五]获取更多技术干货 !


http://www.ppmy.cn/news/1523229.html

相关文章

一次windows server 服务器病毒分析处理总结

接到反应某企业绿盟平台一直报警&#xff0c;当天发现服务器IP地址为192.168.xx.xx 的一天有上千条报警。针对这种情况&#xff0c;进行了分析并进行了处理。 一、绿盟安全管理平台报警 ​ 平台检测到某服务器有远程命令执行漏洞、挖矿行为等异常行为。平台目前最早记录为 20…

Leetcode39- 使数组中所有元素都等于零(2357)

1、题目 给你一个非负整数数组 nums 。在一步操作中&#xff0c;你必须&#xff1a; 选出一个正整数 x &#xff0c;x 需要小于或等于 nums 中 最小 的 非零 元素。 nums 中的每个正整数都减去 x。 返回使 nums 中所有元素都等于 0 需要的 最少 操作数。 示例 1&#xff1a;…

Linux学习-inotify和rsync同步

Inotify 是一个 Linux特性&#xff0c;它监控文件系统操作&#xff0c;比如读取、写入和创建。Inotify 反应灵敏&#xff0c;用法非常简单&#xff0c;并且比 cron 任务的繁忙轮询高效得多。 Inotify实际基于事件驱动机制&#xff0c;为应用程序监控文件系统事件提供了实时响应…

AES加密算法说明

首先&#xff0c;我们得了解AES加密算法的一些基本概念。AES是一种对称加密算法&#xff0c;所谓对称&#xff0c;是说它的加密、解密过程使用相同的密钥。还有非对称加密算法&#xff0c;例如RSA&#xff0c;加密解密使用的是公私钥对。 AES同时是一种分组加密算法&#xff0c…

240909-ChuanhuChatGPT集成Ollama的环境配置

A. 最终效果 B. 需求文件 requirements.txt (至少需要安装这个&#xff0c;具体参见官网)requirements_advanced.txt &#xff08;如果安装了Ollama&#xff0c;并且可以进行对话&#xff0c;可以不需要安装&#xff0c;具体参见官网&#xff09;requirements_succcess.txt&am…

Axure中继器介绍

中继器我们一般在处理重复性比较高的任务时&#xff0c;能让我们达到事半功倍的效果&#xff0c;中继器在整个axure中属于复杂程度比较高的功能&#xff0c;我们今天大致讲一下常用的方法即可。 一、声明一个中继器 默认展示为三行。 点击样式&#xff0c;这里我们可以添加删…

python内置模块datetime.time类详细介绍

​​​​​​​Python的datetime模块是一个强大的日期和时间处理库&#xff0c;它提供了多个类来处理日期和时间。主要包括几个功能类datetime.date、datetime.time、datetime.datetime、datetime.timedelta,datetime.timezone等。 ----------动动小手&#xff0c;非常感谢各位…

C++解决:求排列数

描述 输入两个整数m,n&#xff0c;求m个数字中选n个数的排列数。&#xff08;1<n<m<50&#xff09; 输入描述 两个正整数m和n。 输出描述 一个正整数表示排列数。 用例输入 1 6 5 用例输出 1 720 AC code #include<bits/stdc.h> using namespace s…