前言

在使用spring的过程中,我们有没有发现它的扩展能力很强呢? 由于这个优势的存在,使得spring具有很强的包容性,所以很多第三方应用或者框架可以很容易的投入到spring的怀抱中。今天我们主要来学习Spring中很常用的11个扩展点,你用过几个呢?

1. 类型转换器

如果接口中接收参数的实体对象中,有一个字段类型为Date,但实际传递的参数是字符串类型:2022-12-15 10:20:15,该如何处理?

Spring提供了一个扩展点,类型转换器Type Converter,具体分为3类:

  • Converter<S,T>: 将类型 S 的对象转换为类型 T 的对象
  • ConverterFactory<S, R>: 将 S 类型对象转换为 R 类型或其子类对象
  • GenericConverter:它支持多种源和目标类型的转换,还提供了源和目标类型的上下文。 此上下文允许您根据注释或属性信息执行类型转换。

还是不明白的话,我们举个例子吧。

  1. 定义一个用户对象
@Data
public class User {
private Long id;
private String name;
private Date registerDate;
}
  1. 实现Converter接口
public class DateConverter implements Converter<String, Date> {
private SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
@Override
public Date convert(String source) {
if (source != null && !"".equals(source)) {
try {
simpleDateFormat.parse(source);
} catch (ParseException e) {
e.printStackTrace();
}
}
return null;
}
}
  1. 将新定义的类型转换器注入到Spring容器中
@Configuration
public class WebConfig extends WebMvcConfigurerAdapter {
@Override
public void addFormatters(FormatterRegistry registry) {
registry.addConverter(new DateConverter());
}
}
  1. 调用接口测试
@RequestMapping("/user")
@RestController
public class UserController {
@RequestMapping("/save")
public String save(@RequestBody User user) {
return "success";
}
}

请求接口时,前端传入的日期字符串,会自动转换成Date类型。

2. 获取容器Bean

在我们日常开发中,经常需要从Spring容器中获取bean,但是你知道如何获取Spring容器对象吗?

2.1 BeanFactoryAware

@Service
public class PersonService implements BeanFactoryAware {
private BeanFactory beanFactory; @Override
public void setBeanFactory(BeanFactory beanFactory) throws BeansException {
this.beanFactory = beanFactory;
} public void add() {
Person person = (Person) beanFactory.getBean("person");
}
}

实现BeanFactoryAware接口,然后重写setBeanFactory方法,可以从方法中获取spring容器对象。

2.2 ApplicationContextAware

@Service
public class PersonService2 implements ApplicationContextAware {
private ApplicationContext applicationContext; @Override
public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
this.applicationContext = applicationContext;
} public void add() {
Person person = (Person) applicationContext.getBean("person");
}
}

实现ApplicationContextAware接口,然后重写setApplicationContext方法,也可以通过该方法获取spring容器对象。

2.3 ApplicationListener

@Service
public class PersonService3 implements ApplicationListener<ContextRefreshedEvent> {
private ApplicationContext applicationContext;
@Override
public void onApplicationEvent(ContextRefreshedEvent event) {
applicationContext = event.getApplicationContext();
} public void add() {
Person person = (Person) applicationContext.getBean("person");
}
}

3. 全局异常处理

以往我们在开发界面的时候,如果出现异常,要给用户更友好的提示,例如:

@RequestMapping("/test")
@RestController
public class TestController { @GetMapping("/add")
public String add() {
int a = 10 / 0;
return "su";
}
}

如果不对请求添加接口结果做任何处理,会直接报错:

用户可以直接看到错误信息吗?

这种交互给用户带来的体验非常差。 为了解决这个问题,我们通常在接口中捕获异常:

@GetMapping("/add")
public String add() {
String result = "success";
try {
int a = 10 / 0;
} catch (Exception e) {
result = "error";
}
return result;
}

界面修改后,出现异常时会提示:“数据异常”,更加人性化。

看起来不错,但是有一个问题。

如果只是一个接口还好,但是如果项目中有成百上千个接口,还得加异常捕获代码吗?

答案是否定的,这就是全局异常处理派上用场的地方:RestControllerAdvice

@RestControllerAdvice
public class GlobalExceptionHandler { @ExceptionHandler(Exception.class)
public String handleException(Exception e) {
if (e instanceof ArithmeticException) {
return "data error";
}
if (e instanceof Exception) {
return "service error";
}
retur null;
}
}

方法中处理异常只需要handleException,在业务接口中就可以安心使用,不再需要捕获异常(统一有人处理)。

4. 自定义拦截器

Spring MVC拦截器,它可以获得HttpServletRequestHttpServletResponse等web对象实例。

Spring MVC拦截器的顶层接口是HandlerInterceptor,它包含三个方法:

  • preHandle 在目标方法执行之前执行
  • 执行目标方法后执行的postHandle
  • afterCompletion 在请求完成时执行

为了方便,我们一般继承HandlerInterceptorAdapter,它实现了HandlerInterceptor

如果有授权鉴权、日志、统计等场景,可以使用该拦截器,我们来演示下吧。

  1. 写一个类继承HandlerInterceptorAdapter
public class AuthInterceptor extends HandlerInterceptorAdapter {
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)
throws Exception {
String requestUrl = request.getRequestURI();
if (checkAuth(requestUrl)) {
return true;
}
return false;
}
private boolean checkAuth(String requestUrl) {
return true;
}
}
  1. 将拦截器注册到spring容器中
@Configuration
public class WebAuthConfig extends WebMvcConfigurerAdapter { @Bean
public AuthInterceptor getAuthInterceptor() {
return new AuthInterceptor();
}
@Override
public void addInterceptors(InterceptorRegistry registry) {
registry.addInterceptor(new AuthInterceptor());
}
}
  1. Spring MVC在请求接口时可以自动拦截接口,并通过拦截器验证权限。

5. 导入配置

有时我们需要在某个配置类中引入其他的类,引入的类也加入到Spring容器中。 这时候可以使用注解@Import来完成这个功能。

如果你查看它的源代码,你会发现导入的类支持三种不同的类型。

但是我觉得最好把普通类的配置类和@Configuration注解分开解释,所以列出了四种不同的类型:

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相关的拦截器等扩展点,希望对大家有帮助。

欢迎关注个人公众号——JAVA旭阳

更多学习资料请移步:程序员成神之路

Spring中11个最常用的扩展点,你知道几个?的更多相关文章

  1. Spring中Bean的生命周期及其扩展点

    原创作品,可以转载,但是请标注出处地址http://www.cnblogs.com/V1haoge/p/6106456.html Spring中Bean的管理是其最基本的功能,根据下面的图来了解Spr ...

  2. spring mvc 提供的几个常用的扩展点

    转载 :http://blog.csdn.net/gufachongyang02/article/details/43836105 这是spring3 mvc的核心流程图:   SpirngMVC的第 ...

  3. Spring中Bean管理的常用注解

    在Spring中,主要用于管理bean的注解分为四大类:1.用于创建对象.2.用于给对象的属性注入值.3.用于改变作用的范围.4.用于定义生命周期.这几个在开发中经常接触到,也可以说每天都会遇见.其中 ...

  4. Spring系列14:IoC容器的扩展点

    Spring系列14:IoC容器的扩展点 回顾 知识需要成体系地学习,本系列文章前后有关联,建议按照顺序阅读.上一篇我们详细介绍了Spring Bean的生命周期和丰富的扩展点,没有阅读的强烈建议先阅 ...

  5. Spring 中的几个常用的钩子接口

    1.Aware接口 Awear 这个单词的意思是知道的,所以可以猜想以Awear 结尾的接口意思可以把他知道的东西告诉我们.常用的Awear接口有 ApplicationContextAware和 B ...

  6. Spring源码系列 — BeanDefinition扩展点

    前言 前文介绍了Spring Bean的生命周期,也算是XML IOC系列的完结.但是Spring的博大精深,还有很多盲点需要摸索.整合前面的系列文章,从Resource到BeanDefinition ...

  7. Spring Boot 使用Java代码创建Bean并注册到Spring中

    版权声明:本文为博主原创文章,未经博主允许不得转载. https://blog.csdn.net/catoop/article/details/50558333 声明同一个类下的多个实例: packa ...

  8. Spring点滴十一:Spring中BeanFactoryPostProcessor和BeanPostProcessor区别

    Spring中BeanFactoryPostProcessor和BeanPostProcessor都是Spring初始化bean时对外暴露的扩展点.两个接口从名字看起来很相似,但是作用及使用场景却不同 ...

  9. Spring中常用的23中设计模式

    1.spring 中常用的设计模式有23中  分类  设计模式  创建型 工厂方法模式(FactoryMethod).抽象工厂模式(AbstractFactory).建造者模式(Builder).原型 ...

  10. Spring Boot 中如何使用 Dubbo Activate 扩展点

    摘要: 原创出处 www.bysocket.com 「泥瓦匠BYSocket 」欢迎转载,保留摘要,谢谢! 『 公司的核心竞争力在于创新 – <启示录> 』 继续上一篇:< Spri ...

随机推荐

  1. Java程序设计(二)作业

    题目1:输入一个三位十进制数,对其每一位进行筛选,逆序组合后输出. package test; import java.util.*; public class test2{ public stati ...

  2. vue2.x核心源码深入浅出,我还是去看源码了

    平常的工作就是以vue2.x进行开发,因为我是个实用主义者,以前我就一直觉得,你既然选择了这个框架开发你首先就要先弄懂这玩意怎么用,也就是先熟悉vue语法和各种api,而不是去纠结实现它的原理是什么. ...

  3. 重写 hashcode()真有那么简单嘛?

    万万没想到一个 hashcode() 方法,既然会引出一堆的知识盲区,简直了. 起因: 老八股:为什么重写Equals方法要重写HashCode方法. 大声告诉我为什么,闭着眼睛先把答案背出来,啥?这 ...

  4. Tomcat实战之路

    目录 第一节.安装升级 1.1.linux初始化 1.2.安装 1.3.升级 第二节.配置 2.1.虚拟主机 2.2.默认网站首页路径 2.3.跳转 2.4.配置Tomcat日志 第三节.安全 3.1 ...

  5. python-D2-计算机与编程语言

    计算机五大核心 控制器 计算机的指挥系统,可以控制计算机硬件的整体运行 运算器 实现算术运算和逻辑运算 控制器和运算器结合起来就是cpu,也称为中央处理器,是整个电脑的核心. 存储器 分为两类,非永久 ...

  6. PHP获取两个时间差

    <?php //PHP计算两个时间差的方法 $startdate="2017-12-3 12:00:00"; $enddate="2017-12-4 12:00:0 ...

  7. 银行ATM存取款系统(C语言实现)

    这里使用的运行工具是DEV C++.老铁们一定要看仔细了.是DEV C++ 仅供借鉴:这个是大一时期写的.大四的时候整理了一下(本人C语言学的也不太好).肯定很多不足和存在漏洞的地方.仅供借鉴.仅供借 ...

  8. 34.HyperLinkedModelSerializer详解

    HyperLinkedModelSerializer继承ModelSerializer,只是自动多出了一个url字段,其他都是一样的 不同之处在于使用超链接来表示关联关系而不是主键 默认情况下Hype ...

  9. Pandas常用方法

    数据处理很多需要用到pandas,有两个基本类型:Series表示一维数据,DataFrame表示多维.以下是一些常用方法的整理: pandas.Series 创建 Series pandas.Ser ...

  10. Modbus协议及python库实现

    基础知识 硬件层协议:解决0和1的可靠传输,常有RS232.RS485.CAN.IIC.SPI - 软件层协议:解决传输目的,常有Modbus.TCP/IP.CANopen - 协议优点: Modbu ...