我正在实现BeanFactoryPostProcessor
,我正在尝试提取bean定义类名:
@Component
public class MyFactory implements BeanFactoryPostProcessor{
@Override
public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {
String[] beanDefinitionNames = beanFactory.getBeanDefinitionNames();
for (String name : beanDefinitionNames) {
BeanDefinition beanDefinition = beanFactory.getBeanDefinition(name);
// 'null' when using JavaConfig, 'java.lang.String' when using XML
System.out.println(beanDefinition.getBeanClassName());
}
}
}
当我通过XML配置我的bean时,我得到了没有任何问题的类名:
<bean id="arbitraryString" class="java.lang.String"/>
<bean class="com.test.MyFactory"/>
但是,当我使用JavaConfig并在那里定义一个bean(一个简单的String用于演示目的)时,类名将为null:
@Bean
public String arbitraryString () {
return "the bean definition class name will be null";
}
我已经尝试过搜索这个并且无法理解我是否做错了或者这是预期的行为。我在main方法中没有做任何其他事情,但加载上下文(无论是XML还是配置类)。
答案 0 :(得分:1)
我已经对它进行了一些调查,我想我知道问题是什么(不确定我是对的,还是我正在解释它)。但是,这似乎是个问题:
在JavaConfig类中定义bean会导致spring的初始化。我认为这里的顺序非常重要:
它将获取您的配置类,检索所有@Bean注释方法,然后创建这些对象。
现在我相信&#34; root bean&#34;您要引用的是一个实现类,以防您实例化该类。这看起来很自然,但是在spring XML和spring javaconfig之间存在着悖论。
在xml中:
所有bean都定义为类。它们通过调用该类的构造函数来实例化。
在JavaConfig中:
bean不再是一个独立的bean。它被视为工厂bean。所以bean实际上没有root bean类,它有一个工厂bean和一个工厂方法。您将如何从方法定义中设置根类?您可以将根bean设置为类,但在大多数类中都不会成立。工厂可以返回返回类的任何实现,因此在创建BeanDefinition对象时,没有可用的根bean。
通过将appconfig标记为静态(工厂方法)可以观察到这一点。现在,没有工厂bean,因为在扫描配置类时,它不会被创建。这意味着,spring将使用CGI创建一个作为root bean的实现。但是,这不是您期望的根bean。例如,在我的测试中:
com.*.*.AppConfig$$EnhancerBySpringCGLIB$$1d5cc574
这是我在创建静态bean时得到的输出。
所以总的来说:
JavaConfig就像工厂一样,因此在定义形成时不知道创建的bean。
Xml定义良好,实现类是显式设置的,因此给出了根bean。
需要生成静态bean,因为工厂不存在,但仍需要创建类。
我希望有帮助:)
- Artur
编辑:最后的注释,它的预期行为:)
答案 1 :(得分:1)
带有url = url.replaceAll("[|?*<\":>+\\[\\]/']", "");
的bean由@Bean
和bean定义
定义类为ConfigurationClassBeanDefinitionReader
所以。您可以从其AnnotatedBeanDefinition
FactoryMethodMetadata