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