简单了解BeanFactory和FactoryBean

本文深入解析Spring框架中FactoryBean的作用及其实现方式,探讨如何利用FactoryBean创建bean,以及其在容器中的注册与获取机制。文章揭示了FactoryBean在初始化过程中如何在容器中注册两个bean的细节,以及如何正确获取由FactoryBean返回的对象。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

BeanFactory是spring bean容器的根接口,BeanFactory提供了,获取bean,判断是否包含bean,判断单例或者原型,获取bean类型,获取bean别名的api。
我们常见的AnnotationConfigApplicationContext、FileSestemXmlApplicationContext、ClassPathXmlAplicationContext都继承了BeanFactory

在这里插入图片描述

FactoryBean
FactoryBean是spring提供创建bean的一个接口
通过代码来实现一下

public class TempDaoFactoryBean {
    public  void test(){
        System.out.println("TempDaoFactoryBean--test");
    }
}
@Component("daoFactoryBean")
public class DaoFactoryBean implements FactoryBean {

    public  void testBean(){
        System.out.println("testBean------DaoFactoryBean");
    }
    @Override
    public Object getObject() throws Exception {
        return new TempDaoFactoryBean();
    }

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

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

DaoFactoryBean 实现了FactoryBean,并实现了其3个方法,getObject返回的是TempDaoFactoryBean

public class Test {

    public static void main(String[] args) throws Exception {
        AnnotationConfigApplicationContext annotationConfigApplicationContext = new AnnotationConfigApplicationContext(DaoFactoryBean.class);
        DaoFactoryBean daoFactoryBean = (DaoFactoryBean) annotationConfigApplicationContext.getBean("daoFactoryBean");
        daoFactoryBean.testBean();
    }

}

在测试中发现强制转换失败
在这里插入图片描述这里发现我们getBean方法获取名为daoFactoryBean的bean的类型竟然是TempDaoFactoryBean
通过测试发现,我们实现了FactoryBean,在初始化时会在容器中加入2个bean,一个是我们指定的name ,这里就是daoFactoryBean,指定的名字类型是getObject方法返回类的类型。而我们的DaoFactoryBean这个类实例,在容器中的名字为&daoFactoryBean
下面用代码来测试一下

public class Test {

    public static void main(String[] args) throws Exception {
        AnnotationConfigApplicationContext annotationConfigApplicationContext = new AnnotationConfigApplicationContext(DaoFactoryBean.class);
        DaoFactoryBean daoFactoryBean = (DaoFactoryBean) annotationConfigApplicationContext.getBean("&daoFactoryBean");
        daoFactoryBean.testBean();
        TempDaoFactoryBean tempDaoFactoryBean = (TempDaoFactoryBean) annotationConfigApplicationContext.getBean("daoFactoryBean");
        tempDaoFactoryBean.test();
    }

}

测试结果

在这里插入图片描述

spring在获取beanName,是通过name来获取的,但是需要对name进行转换,原因有二,一是name可能以 & 字符开头,二是可能设置了别名
name以&开头,以及转换成设

在这里插入图片描述
在这里插入图片描述

在BeanFactory有一个注释,这个注释FactoryBean返回的不是自己,而是返回的getObject中的,而要拿到FactoryBean需要带上&

在这里插入图片描述在DefaultListableBeanFactory这个类中第734行到761行有这样的代码描述

// Trigger initialization of all non-lazy singleton beans...
//触发非懒加载的单例bean的初始化
		for (String beanName : beanNames) {
			RootBeanDefinition bd = getMergedLocalBeanDefinition(beanName);
			if (!bd.isAbstract() && bd.isSingleton() && !bd.isLazyInit()) {
				if (isFactoryBean(beanName)) {//判断是否为FactoryBean
					Object bean = getBean(FACTORY_BEAN_PREFIX + beanName);//如果是就加上 & 符号
					if (bean instanceof FactoryBean) {
						FactoryBean<?> factory = (FactoryBean<?>) bean;
						boolean isEagerInit;
						if (System.getSecurityManager() != null && factory instanceof SmartFactoryBean) {
							isEagerInit = AccessController.doPrivileged(
									(PrivilegedAction<Boolean>) ((SmartFactoryBean<?>) factory)::isEagerInit,
									getAccessControlContext());
						}
						else {
							isEagerInit = (factory instanceof SmartFactoryBean &&
									((SmartFactoryBean<?>) factory).isEagerInit());
						}
						if (isEagerInit) {
							getBean(beanName);
						}
					}
				}
				else {
					getBean(beanName);
				}
			}
		}

为什么返回的是getObject的对象,当我们使用getBean(Class requiredType)时,有个transfBeanName方法,会截取&。

protected <T> T doGetBean(
			String name, @Nullable Class<T> requiredType, @Nullable Object[] args, boolean typeCheckOnly)
			throws BeansException {
		/**
		 *这里通过name进行转换,获取到beanName
		 * 这里为什么要进行转换,而不直接拿name作为beanName,有两个原因
		 * 第一是:name可能会以 & 这个字符开头
		 * 第二是:别名
		 */
		String beanName = transformedBeanName(name);
		..........
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值