0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

Spring中11个最常用的扩展点分享2

jf_78858299 来源:JAVA旭阳 作者:JAVA旭阳 2023-05-11 10:48 次阅读

5.1 通用类

这种引入方式是最简单的,引入的类会被实例化为一个bean对象。

public class A {
}

@Import(A.class)
@Configuration
public class TestConfiguration {

}

通过@Import注解引入类A,spring可以自动实例化A对象,然后在需要使用的地方通过注解@Autowired注入:

@Autowired
private A a;

5.2 配置类

这种引入方式是最复杂的,因为@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所在的属性。

5.3 ImportSelector

该导入方法需要实现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方法返回的是一个数组,也就是说可以同时引入多个类,非常方便。

5.4 ImportBeanDefinitionRegistrar

该导入方法需要实现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种。

6. 当工程启动时

有时候我们需要在项目启动的时候自定义一些额外的功能,比如加载一些系统参数,完成初始化,预热本地缓存等。我们应该做什么?

好消息是 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注解来指定。

7. 修改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的相关对象,修改对象的属性。

8. 初始化 Bean 前和后

有时,您想在 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等注解都是通过AutowiredAnnotationBeanPostProcessorCommonAnnotationBeanPostProcessor来实现的。

9. 初始化方法

目前在Spring中初始化bean的方式有很多种:

  1. 使用@PostConstruct注解
  2. 实现InitializingBean接口

9.1 使用 @PostConstruct

@Service
public class AService {
    @PostConstruct
    public void init() {
        System.out.println("===init===");
    }
}

为需要初始化的方法添加注解@PostConstruct,使其在Bean初始化时执行。

9.2 实现初始化接口InitializingBean

@Service
public class BService implements InitializingBean {

    @Override
    public void afterPropertiesSet() throws Exception {
        System.out.println("===init===");
    }
}

实现InitializingBean接口,重写afterPropertiesSet方法,在该方法中可以完成初始化功能。

10. 关闭Spring容器前

有时候,我们需要在关闭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方法做一些额外的工作。

通常我们会同时实现InitializingBeanDisposableBean接口,重写初始化方法和销毁方法。

11. 自定义Beanscope

我们都知道spring core默认只支持两种Scope

  • Singleton单例,从spring容器中获取的每一个bean都是同一个对象。
  • prototype多实例,每次从spring容器中获取的bean都是不同的对象。

Spring Web 再次扩展了 Scope,添加

  • RequestScope:同一个请求中从spring容器中获取的bean都是同一个对象。
  • SessionScope:同一个session从spring容器中获取的bean都是同一个对象。

尽管如此,有些场景还是不符合我们的要求。

比如我们在同一个线程中要从spring容器中获取的bean都是同一个对象,怎么办?

答案 :这需要一个自定义范围。

  1. 实现 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;
    }
}
  1. 将新定义的Scope注入到Spring容器中
@Component
public class ThreadLocalBeanFactoryPostProcessor implements BeanFactoryPostProcessor {
    @Override
    public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {
        beanFactory.registerScope("threadLocalScope", new ThreadLocalScope());
    }
}
  1. 使用新定义的Scope
@Scope("threadLocalScope")
@Service
public class CService {
    public void add() {
    }
}

总结

本文总结了Spring中很常用的11个扩展点,可以在Bean创建、初始化到销毁各个阶段注入自己想要的逻辑,也有Spring MVC相关的拦截器等扩展点,希望对大家有帮助。

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 转换器
    +关注

    关注

    27

    文章

    8207

    浏览量

    141851
  • 框架
    +关注

    关注

    0

    文章

    297

    浏览量

    17045
  • spring
    +关注

    关注

    0

    文章

    332

    浏览量

    14161
收藏 人收藏

    评论

    相关推荐

    什么是java spring

    的事情。然而,Spring的用途不仅限于服务器端的开发。从简单性、可测试性和松耦合的角度而言,任何Java应用都可以从Spring受益。Spring是一
    发表于 09-11 11:16

    电脑高手最常用的五组合键

    电脑高手最常用的五组合键电脑高手最常用的五组合键winkey+d :这是高手最常用的第一快捷组合键。这个快捷键组合可以将桌面上的所有窗口
    发表于 02-11 12:34

    分享两开发过程最常用的文件

    分享两开发过程最常用的文件
    发表于 10-25 14:23

    聊聊Dubbo - Dubbo可扩展机制源码解析

    的ExtensionLoader实体类。通过这个实体类,可以根据name获得具体的扩展,也可以获得一自适应扩展。*2. getExtension方法getExtention方法
    发表于 06-05 18:43

    Spring工作原理

    Spring所提供的这些服务和功能。Spring里用的最经典的一设计模式就是:模板方法模式。(这里我都不介绍了,是一常用的设计模式)
    发表于 07-10 07:41

    最常用的Linux命令盘点

    玩过Linux的人都会知道,Linux的命令的确是非常多,但是玩过Linux的人也从来不会因为Linux的命令如此之多而烦恼,因为我们只需要掌握我们最常用的命令就可以了。当然你也可以在使用时去找
    发表于 07-11 08:21

    启动Spring Boot项目应用的三种方法

    基础。我们知道了Spring Boot是什么了,那么我们又该如何启动Spring Boot应用呢?这里小编给大家推荐常用的三种方法。分别是IDEA编辑器启动、命令启动、java命令j
    发表于 01-14 17:33

    STM32时钟系统最常用的知识

    ,讲一最常用的知识。其它的型号大同小异。时钟含义特点HSE外部高速时钟信号一般选择外接晶振,最常用的时钟信号。电机驱动板外接12MhzHSI内部高速时钟信号由单片...
    发表于 08-11 07:39

    最常用11款Kubernetes工具

    助于改善 Kubernetes 开发人员的体验。本文列出作者自己最常用11 款 Kubernetes 工具,并对它们进行了分类介绍。 Kubernetes 是一个非常强大的容器编排平台。但在
    的头像 发表于 08-23 10:43 1896次阅读

    Spring最常用11扩展

    除此之外,我们在使用spring的过程中,有没有发现它的扩展能力非常强。由于这个优势的存在,让spring拥有强大的包容能力,让很多第三方应用能够轻松投入spring的怀抱。比如:ro
    的头像 发表于 01-11 10:31 572次阅读

    剖析Spring最常用扩展点(上)

    我们一说到spring,可能第一个想到的是 `IOC`(控制反转) 和 `AOP`(面向切面编程)。 没错,它们是spring的基石,得益于它们的优秀设计,使得spring能够从众多优秀框架中脱颖而出。
    的头像 发表于 02-15 16:06 472次阅读
    剖析<b class='flag-5'>Spring</b>中<b class='flag-5'>最常用</b>的<b class='flag-5'>扩展</b>点(上)

    剖析Spring最常用扩展点(中)

    我们一说到spring,可能第一个想到的是 `IOC`(控制反转) 和 `AOP`(面向切面编程)。 没错,它们是spring的基石,得益于它们的优秀设计,使得spring能够从众多优秀框架中脱颖而出。
    的头像 发表于 02-15 16:06 321次阅读
    剖析<b class='flag-5'>Spring</b>中<b class='flag-5'>最常用</b>的<b class='flag-5'>扩展</b>点(中)

    剖析Spring最常用扩展点(下)

    我们一说到spring,可能第一个想到的是 `IOC`(控制反转) 和 `AOP`(面向切面编程)。 没错,它们是spring的基石,得益于它们的优秀设计,使得spring能够从众多优秀框架中脱颖而出。
    的头像 发表于 02-15 16:07 286次阅读

    基于spring的SPI扩展机制是如何实现的?

    基本上,你一说是基于 spring 的 SPI 扩展机制,再把spring.factories文件和EnableAutoConfiguration提一下,那么这个问题就答的八九不离十了。
    的头像 发表于 03-07 09:17 699次阅读

    Spring11最常用扩展点分享1

    在使用spring的过程中,我们有没有发现它的扩展能力很强呢?由于这个优势的存在,使得spring具有很强的包容性,所以很多第三方应用或者框架可以很容易的投入到spring的怀抱中。今
    的头像 发表于 05-11 10:48 370次阅读