电子说
这种引入方式是最简单的,引入的类会被实例化为一个bean对象。
public class A {
}
@Import(A.class)
@Configuration
public class TestConfiguration {
}
通过@Import
注解引入类A,spring可以自动实例化A对象,然后在需要使用的地方通过注解@Autowired
注入:
@Autowired
private A a;
这种引入方式是最复杂的,因为@Configuration支持还支持多种组合注解,比如:
@Import
@ImportResource
@PropertySource
public class A {
}
public class B {
}
@Import(B.class)
@Configuration
public class AConfiguration {
@Bean
public A a() {
return new A();
}
}
@Import(AConfiguration.class)
@Configuration
public class TestConfiguration {
}
@Configuration
注解的配置类通过@Import
注解导入,配置类@Import
、@ImportResource
相关注解引入的类会一次性全部递归引入@PropertySource
所在的属性。
该导入方法需要实现ImportSelector
接口
public class AImportSelector implements ImportSelector {
private static final String CLASS_NAME = "com.sue.cache.service.test13.A";
public String[] selectImports(AnnotationMetadata importingClassMetadata) {
return new String[]{CLASS_NAME};
}
}
@Import(AImportSelector.class)
@Configuration
public class TestConfiguration {
}
这种方法的好处是selectImports
方法返回的是一个数组,也就是说可以同时引入多个类,非常方便。
该导入方法需要实现ImportBeanDefinitionRegistrar
接口:
public class AImportBeanDefinitionRegistrar implements ImportBeanDefinitionRegistrar {
@Override
public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {
RootBeanDefinition rootBeanDefinition = new RootBeanDefinition(A.class);
registry.registerBeanDefinition("a", rootBeanDefinition);
}
}
@Import(AImportBeanDefinitionRegistrar.class)
@Configuration
public class TestConfiguration {
}
这种方法是最灵活的。容器注册对象可以在registerBeanDefinitions
方法中获取,可以手动创建BeanDefinition
注册到BeanDefinitionRegistry
种。
有时候我们需要在项目启动的时候自定义一些额外的功能,比如加载一些系统参数,完成初始化,预热本地缓存等。我们应该做什么?
好消息是 SpringBoot 提供了:
CommandLineRunner
ApplicationRunner
这两个接口帮助我们实现了上面的需求。
它们的用法很简单,以ApplicationRunner
接口为例:
@Component
public class TestRunner implements ApplicationRunner {
@Autowired
private LoadDataService loadDataService;
public void run(ApplicationArguments args) throws Exception {
loadDataService.load();
}
}
实现ApplicationRunner
接口,重写run
方法,在该方法中实现您的自定义需求。
如果项目中有多个类实现了ApplicationRunner
接口,如何指定它们的执行顺序?
答案是使用@Order(n)注解,n的值越小越早执行。当然,顺序也可以通过@Priority
注解来指定。
BeanDefinition
在实例化Bean对象之前,Spring IOC
需要读取Bean的相关属性,保存在BeanDefinition
对象中,然后通过BeanDefinition
对象实例化Bean
对象。
如果要修改BeanDefinition对象中的属性怎么办?
答案 :我们可以实现 BeanFactoryPostProcessor
接口。
@Component
public class MyBeanFactoryPostProcessor implements BeanFactoryPostProcessor {
@Override
public void postProcessBeanFactory(ConfigurableListableBeanFactory configurableListableBeanFactory) throws BeansException {
DefaultListableBeanFactory defaultListableBeanFactory = (DefaultListableBeanFactory) configurableListableBeanFactory;
BeanDefinitionBuilder beanDefinitionBuilder = BeanDefinitionBuilder.genericBeanDefinition(User.class);
beanDefinitionBuilder.addPropertyValue("id", 123);
beanDefinitionBuilder.addPropertyValue("name", "Tom");
defaultListableBeanFactory.registerBeanDefinition("user", beanDefinitionBuilder.getBeanDefinition());
}
}
在postProcessBeanFactory
方法中,可以获取BeanDefinition
的相关对象,修改对象的属性。
有时,您想在 bean 初始化前后实现一些您自己的逻辑。
这时候就可以实现:BeanPostProcessor
接口。
该接口目前有两个方法:
postProcessBeforeInitialization
:应该在初始化方法之前调用。postProcessAfterInitialization
:此方法在初始化方法之后调用。@Component
public class MyBeanPostProcessor implements BeanPostProcessor {
@Override
public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
if (bean instanceof User) {
((User) bean).setUserName("Tom");
}
return bean;
}
}
我们经常使用的@Autowired
、@Value
、@Resource
、@PostConstruct
等注解都是通过AutowiredAnnotationBeanPostProcessor
和CommonAnnotationBeanPostProcessor
来实现的。
目前在Spring中初始化bean的方式有很多种:
@PostConstruct
注解InitializingBean
接口@PostConstruct
@Service
public class AService {
@PostConstruct
public void init() {
System.out.println("===init===");
}
}
为需要初始化的方法添加注解@PostConstruct
,使其在Bean初始化时执行。
InitializingBean
@Service
public class BService implements InitializingBean {
@Override
public void afterPropertiesSet() throws Exception {
System.out.println("===init===");
}
}
实现InitializingBean
接口,重写afterPropertiesSet
方法,在该方法中可以完成初始化功能。
有时候,我们需要在关闭spring容器之前做一些额外的工作,比如关闭资源文件。
此时你可以实现 DisposableBean
接口并重写它的 destroy
方法。
@Service
public class DService implements InitializingBean, DisposableBean {
@Override
public void destroy() throws Exception {
System.out.println("DisposableBean destroy");
}
@Override
public void afterPropertiesSet() throws Exception {
System.out.println("InitializingBean afterPropertiesSet");
}
}
这样,在spring容器销毁之前,会调用destroy
方法做一些额外的工作。
通常我们会同时实现InitializingBean
和DisposableBean
接口,重写初始化方法和销毁方法。
Bean
的scope
我们都知道spring core默认只支持两种Scope
:
Singleton
单例,从spring容器中获取的每一个bean都是同一个对象。prototype
多实例,每次从spring容器中获取的bean都是不同的对象。Spring Web 再次扩展了 Scope,添加
RequestScope
:同一个请求中从spring容器中获取的bean都是同一个对象。SessionScope
:同一个session从spring容器中获取的bean都是同一个对象。尽管如此,有些场景还是不符合我们的要求。
比如我们在同一个线程中要从spring
容器中获取的bean
都是同一个对象,怎么办?
答案 :这需要一个自定义范围。
Scope
接口public class ThreadLocalScope implements Scope {
private static final ThreadLocal THREAD_LOCAL_SCOPE = new ThreadLocal();
@Override
public Object get(String name, ObjectFactory? objectFactory) {
Object value = THREAD_LOCAL_SCOPE.get();
if (value != null) {
return value;
}
Object object = objectFactory.getObject();
THREAD_LOCAL_SCOPE.set(object);
return object;
}
@Override
public Object remove(String name) {
THREAD_LOCAL_SCOPE.remove();
return null;
}
@Override
public void registerDestructionCallback(String name, Runnable callback) {
}
@Override
public Object resolveContextualObject(String key) {
return null;
}
@Override
public String getConversationId() {
return null;
}
}
@Component
public class ThreadLocalBeanFactoryPostProcessor implements BeanFactoryPostProcessor {
@Override
public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {
beanFactory.registerScope("threadLocalScope", new ThreadLocalScope());
}
}
@Scope("threadLocalScope")
@Service
public class CService {
public void add() {
}
}
本文总结了Spring中很常用的11个扩展点,可以在Bean创建、初始化到销毁各个阶段注入自己想要的逻辑,也有Spring MVC相关的拦截器等扩展点,希望对大家有帮助。
全部0条评论
快来发表一下你的评论吧 !