FactoryBean 原理简介

FactoryBean 首先是一个工厂类,它可以生产指定的Bean,特殊之处在于它可以向Spring容器中注册两个Bean,一个是它本身,一个是FactoryBean.getObject()方法返回值所代表的Bean。通过实现 FactoryBean 接口,你可以控制某个 Bean 的实例化过程,提供比默认机制更复杂的创建逻辑。

FactoryBean接口

public interface FactoryBean<T> {@NullableT getObject() throws Exception;@NullableClass<?> getObjectType();default boolean isSingleton() {return true;}
}

FactoryBean 接口包含三个核心方法:

  • Object getObject() throws Exception
    返回由 FactoryBean 创建的 bean 实例。这是 FactoryBean 最重要的方法,通过这个方法,你可以自定义 bean 的创建逻辑。
  • Class<?> getObjectType()
    返回由 FactoryBean 创建的 bean 实例的类型。这个方法用于告诉 Spring 容器这个工厂 bean 创建的对象的类型,以便在需要时进行类型转换和检查。
  • boolean isSingleton()
    返回由 FactoryBean 创建的 bean 实例是否是单例的。如果返回 true,那么 Spring 容器会将创建的对象作为单例进行管理;如果返回 false,每次请求都会创建一个新的实例。

重要事项

  • FactoryBean表现的是一个工厂的职责。 即一个Bean A如果实现了FactoryBean接口,那么A就变成了一个工厂,根据A的名称获取到的实际上是工厂调用getObject()返回的对象,而不是A本身,如果要获取工厂A自身的实例,那么需要在名称前面加上’&'符号。详情请看实例演示。

实例演示

定义学生类

注意这个类并没有被@Component等注解标注,即没有被Spring容器管理。

public class Student {private Logger logger = LoggerFactory.getLogger(Student.class);public Student(){logger.info("Hi, I am a good student!");}
}
定义StudentFactoryBean
package com.example.jaytecharchite.factorybeandemo;
@Component
public class StudentFactoryBean implements FactoryBean {@Overridepublic Object getObject() throws Exception {return new Student();}@Overridepublic Class<?> getObjectType() {return Student.class;}@Overridepublic boolean isSingleton() {// return false;  非单例return FactoryBean.super.isSingleton(); //默认true,单例工厂}
}

很简单,我们让这个工厂类生成学生类,注意看getObjectType()返回的Class类型和getObject()返回的实例一致,其实也可以不一致,当我们问Spring容器通过Student student = applicationContext.getBean(Student.class);这一句代码向Spring容器获取Bean时,其实它是通过getObjectType()来找到生产这个类型的工厂,从而调用getObject()获取到实例。

配置类

需要配置类将StudentFactoryBean注册到Spring容器中。

@Configuration
@ComponentScan("com.example.jaytecharchite")
public class AppConfig {
}
测试
@SpringBootTest
public class MainTest {private Logger logger = LoggerFactory.getLogger(CallBackTest.class);@Testpublic void test(){AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext(AppConfig.class);logger.info("容器启动完成!");Student student = applicationContext.getBean(Student.class);System.out.println(student);Student student2 = applicationContext.getBean(Student.class);System.out.println(student2);Object student3 = applicationContext.getBean("studentFactoryBean");System.out.println(student3);Object customerFactoryBean2 = applicationContext.getBean("&studentFactoryBean");System.out.println(customerFactoryBean2);}
}

输出

如果isSingleton()返回的是true,则输出结果如下:

2024-07-03 22:37:38.961 INFO 21112 — [ main] c.e.j.xxxaware.CallBackTest : 容器启动完成!
2024-07-03 22:37:38.962 INFO 21112 — [ main] c.e.j.factorybeandemo.Student : Hi, I am a good student!
com.example.jaytecharchite.factorybeandemo.Student@5c887052
com.example.jaytecharchite.factorybeandemo.Student@5c887052
com.example.jaytecharchite.factorybeandemo.Student@5c887052
com.example.jaytecharchite.factorybeandemo.StudentFactoryBean@55fdf7f9

可以看见前三个获取到的Bean都是getObejct()方法的Bean,并且是同一个Bean,因为是设置了单例模式。而最后一个使用了&符号获取到的才是 StudentFactoryBean本身的实例。

如果isSingleton()返回的是false,则输出结果如下:

2024-07-03 22:40:58.104 INFO 4416 — [ main] c.e.j.xxxaware.CallBackTest : 容器启动完成!
2024-07-03 22:40:58.105 INFO 4416 — [ main] c.e.j.factorybeandemo.Student : Hi, I am a good student!
com.example.jaytecharchite.factorybeandemo.Student@72b6832e
2024-07-03 22:40:58.105 INFO 4416 — [ main] c.e.j.factorybeandemo.Student : Hi, I am a good student!
com.example.jaytecharchite.factorybeandemo.Student@3850e90c
2024-07-03 22:40:58.105 INFO 4416 — [ main] c.e.j.factorybeandemo.Student : Hi, I am a good student!
com.example.jaytecharchite.factorybeandemo.Student@3d9f5016
com.example.jaytecharchite.factorybeandemo.StudentFactoryBean@7e91ed74

可以看见前三个Bean都是不同的对象实例,并且每个实例都会执行一次构造方法。我们自定义的StudentFactoryBean实现了FactoryBean接口,所以当StudentFactoryBean被扫描进Spring容器时,实际上它向容器中注册了两个bean,一个是StudentFactoryBean类的单例对象;另外一个就是getObject()方法返回的对象,在demo中,我们重写的getObject()方法中,我们通过new Student()返回了一个Student的实例对象,所以我们从容器中能获取到Student的实例对象。如果我们想通过beanName去获取StudentFactoryBean的单例对象,需要在beanName前面添加一个&符号,这样就能根据beanName获取到原生对象了。否则获取到的还是getObject()提供的对象。

FactoryBean 的使用场景

FactoryBean 的使用场景包括但不限于:

  • 创建复杂对象:
    当对象的创建过程非常复杂,无法通过简单的构造函数或静态工厂方法实现时,可以使用 FactoryBean。例如,创建带有复杂初始化逻辑的数据库连接对象、远程服务代理等。
  • 动态代理:
    使用 FactoryBean 可以方便地创建动态代理对象,特别是在 AOP(面向切面编程)和远程调用场景中。
  • 单例和多例模式:
    通过 isSingleton 方法,可以灵活地控制 bean 的作用域。

神级现场

在MyBatis中,只需要写个接口就能实现就能运行SQL你不觉得奇怪吗?难道你的接口MyBatis自动帮你实现了?非也,MyBatis并没有去实现你的接口!那么你会问,那为什么直接可以调用接口?这就是FactoryBean+Proxy的神级现场设计,下面为了简答只演示调用接口就实现功能。

学生接口
public interface Student {void say();
}

代理工厂Bean


/*** 这里根本没有Student的实现类,只有一个接口,在代理中实现了接口的方法* @param <T>*/
@Component
public class StudentFactoryBeanProxy<T> implements FactoryBean<T> {/*** 通过 FactoryBean 创建代理对象Bean* @return 一个代理对象* @throws Exception*/@Overridepublic T getObject() throws Exception {// 创建了一个 InvocationHandler 对象,用于处理代理对象的方法调用InvocationHandler handler = (proxy, method, args) -> {String name = method.getName();if ("say".equals(name)) {System.out.println("Hi, I am a good student!");}return "只要拿到接口,就能在代理中实现任何功能!";};// 使用 Proxy.newProxyInstance() 方法创建代理对象,也就是外部的接口其实是一个代理对象return (T) Proxy.newProxyInstance(Thread.currentThread().getContextClassLoader(), new Class[]{Student.class}, handler);}@Overridepublic Class<?> getObjectType() {return Student.class;}
}

配置类

@Configuration
@ComponentScan("com.example.jaytecharchite.factorybeandemo2")
public class MyConfig {
}

测试

public class MainTest {private Logger logger = LoggerFactory.getLogger(MainTest.class);@Autowiredprivate Student student; // 其实这个就是一个代理对象,只不过名字还是Student@Testpublic void test(){AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext(MyConfig.class);logger.info("容器启动完成!");student.say();}
}

输出:

2024-07-03 23:24:36.144 INFO 17080 — [ main] c.e.j.factorybeandemo2.MainTest : 容器启动完成!
Hi, I am a good student!

注意看,我们并没有实现Student哦,但是它调用student.say();就能说话!神级现场!!!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/39145.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

线程和进程的区别及应用场景

线程和进程的区别及应用场景 大家好&#xff0c;我是免费搭建查券返利机器人省钱赚佣金就用微赚淘客系统3.0的小编&#xff0c;也是冬天不穿秋裤&#xff0c;天冷也要风度的程序猿&#xff01;今天我们将深入探讨计算机中两个重要的概念&#xff1a;线程和进程。这两者在操作系…

人工智能--循环神经网络

个人主页&#xff1a;欢迎来到 Papicatch的博客 课设专栏 &#xff1a;学生成绩管理系统 专业知识专栏&#xff1a; 专业知识 文章目录 &#x1f349;引言 &#x1f349;概述 &#x1f348;基本概念 &#x1f34d;定义 &#x1f34d;结构 &#x1f34c;输入层 &#…

行业模板|DataEase旅游行业大屏模板推荐

DataEase开源数据可视化分析工具于2022年6月发布模板市场&#xff08;https://templates-de.fit2cloud.com&#xff09;&#xff0c;并于2024年1月新增适用于DataEase v2版本的模板分类。模板市场旨在为DataEase用户提供专业、美观、拿来即用的大屏模板&#xff0c;方便用户根据…

探索编程世界的奇妙之旅

在这个数字化时代&#xff0c;编程&#xff0c;这一曾经看似高深莫测的技能&#xff0c;正逐渐渗透到我们生活的方方面面。从智能手机上的应用程序&#xff0c;到自动驾驶的汽车&#xff0c;再到支撑互联网运行的庞大服务器集群&#xff0c;背后都离不开编程的力量。今天&#…

【CSharp】使用enum枚举编程的优点

【CSharp】使用enum枚举编程的优点 1.背景2.代码1.背景 在程序开发中为什么会使用enum枚举呢,其实不使用enum代码也能跑, 但是随着业务越来越复杂,年代久远,那也许就只能你能跑。 所谓面向bug编程,来获得不可取代地位的思维逻辑,在AI时代洪流面前,从一开始就将错付。 …

开关电源的电路组成原理

一、 开关电源的电路组成 开关电源的主要电路是由输入电磁干扰滤波器&#xff08;EMI&#xff09;、整流滤波电路、功率变换电路、PWM控制器电路、输出整流滤波电路组成。辅助电路有输入过欠压保护电路、输出过欠压保护电路、输出过流保护电路、输出短路保护电路等。 开关电源…

【语音识别】传统语音识别算法概述,应用场景,项目实践及案例分析,附带代码示例

传统语音识别算法是将语音信号转化为文本形式的技术&#xff0c;它主要基于模式识别理论和数学统计学方法。以下是传统语音识别算法的基本概述&#xff1a; 1. 基本原理 传统语音识别算法包括以下几个关键步骤&#xff1a; 预处理&#xff1a;将原始语音信号进行采样、滤波、…

基于Hadoop平台的电信客服数据的处理与分析④项目实现:任务18: 数据展示

任务描述 接下来我们需要将根据业务需求将某人按照不同维度查询出来的结果&#xff0c;展示到Web页面上。 任务指导 数据展示模块流程图&#xff1a; 数据展示使用Java的SSM框架&#xff0c;需要实现的代码包括&#xff1a; 1. 实体类 2. 数据库操作 3. 业务逻辑操作 4.…

新疆水博会将举办多场高端论坛探析水利科技创新发展

由新疆维吾尔自治区水利学会主办的第三届新疆国际水利科技博览会暨新疆水利科技创新发展论坛&#xff0c;将于2024年8月8日至9日在新疆国际会展中心召开&#xff0c;同期将举办第三届新疆国际供排水及智慧水务技术设备展览会。 据悉&#xff0c;新疆水博会期间将举办新疆水利科…

springboot旅游管理系统-计算机毕业设计源码16021

摘 要 本文旨在设计和实现一个基于Spring Boot框架的旅游管理系统。该系统通过利用Spring Boot的快速开发特性和丰富的生态系统&#xff0c;提供了一个高效、可靠和灵活的解决方案。系统将实现旅游景点信息的管理、线路规划、跟团游玩、旅游攻略、酒店信息管理、订单管理和用户…

MySQL—创建和修改数据表结构

创建表 实例&#xff1a; CREATE TABLE user (id INT,name VARCHAR(255),password VARCHAR(255),birthday DATE) CHARACTER SET utf8 COLLATE utf8_bin ENGINE INNODB; 显示数据库中的表 show tables from hsp; 显示表结构 desc dept; 修改表 实例&#xff1a; 代码&…

Rust破界:前端革新与Vite重构的深度透视(中)

Rust破界&#xff1a;前端革新与Vite重构的深度透视 Rust 重构 Vite 的深度剖析技术瓶颈与 Rust 的解法实例分析&#xff1a;性能跃升的实践 Rust 在前端工具链的广泛影响从 Vite 到更广阔的舞台成功案例&#xff1a;其他前端项目的 Rust 实践技术动因与行业趋势多样性思考&…

第十五章 路由器综合路由配置

实验目标 掌握综合路由器的配置方法&#xff1b; 掌握查看通过路由重分布学习产生的路由&#xff1b; 熟悉广域网线缆的链接方式&#xff1b; 实验背景 假设某公司通过一台三层交换机连到公司出口路由器 R1 上&#xff0c;路由器 R1 再和公司外的另一台路由器 R2 连接。…

C++中using关键字介绍

C中using关键字介绍 C中using关键字有两种用法&#xff0c;using 指令&#xff08;Using Directive&#xff09;和using 声明&#xff08;Using Declaration&#xff09; using 指令影响整个命名空间&#xff0c;using 声明只影响特定名称。 using 指令 定义&#xff1a;usi…

【Python实战因果推断】17_线性回归的不合理效果7

目录 Regression for Dummies Conditionally Random Experiments Dummy Variables Regression for Dummies 回归和正交化固然很好&#xff0c;但归根结底&#xff0c;你必须做出独立性假设。你必须假设&#xff0c;在考虑到某些协变量的情况下&#xff0c;干预看起来与随机分…

k8s 常用的命令

k8s 常用的操作 查找资源 kubectl get&#xff1a; 获取所有的资源&#xff0c;包括node、namespace、pod 、service、deployment等&#xff0c;可以展示一个或者多个资源。 创建资源 kubectl create &#xff1a;Kubernetes 的清单文件可以用 json 或 yaml 定义。 更新资源 …

SQL Error: 1054, SQLState: 42S22

SQL 错误 1054 通常与 SQL 查询中的未知列有关&#xff0c;SQLState 42S22表示列未找到错误。 解决方式&#xff1a; 检查列名&#xff1a; 确保您在SQL查询中使用的列名实际存在于您查询的表中。可能存在拼写错误或列名错误。验证表名&#xff1a; 确认SQL查询中的表名是否正…

python 获取Shopee虾皮商家店铺商品列表 虾皮api数据采集

此api接口可用于获取虾皮平台商家店铺的商品列表&#xff0c;目前land参数支持id、vn、my、th、sg、ph、tw&#xff08;印尼、越南、马来、泰国、新加坡、菲律宾、台湾&#xff09;。 若有需要&#xff0c;请点击文末链接联系我们。 详细采集页面如下 https://shopee.tw/yue…

使用Adobe Acrobat对PDF文档进行数字签名

文章目录 前言一、使用Adobe Acrobat对PDF文档进行数字签名1.使用Adobe Acrobat打开需要进行签名的PDF文档2. 点击【查看更多】3.点击【使用证书】4.点击【数字签名】5.使用鼠标选定一个区域6.选择您需要使用的证书 → 点击【继续】7.点击【签名】8.签名成功 前言 一、使用Ado…

嵌入式C语言中指针与链表的关系详解

假定给你一块非常小的内存,这块内存只有8字节,这里也没有高级语言,没有操作系统,你操作的数据单位是单个字节,你该怎样读写这块内存呢? 注意这里的限定,再读一遍,没有高级语言,没有操作系统,在这样的限制之下,你必须直面内存读写的本质。 这个本质是什么呢? 本质…