BeanPostProcessor简介

BeanPostProcessor是Spring IOC容器给我们提供的一个扩展接口。接口声明如下:

  1. public interface BeanPostProcessor {
  2. //bean初始化方法调用前被调用
  3. Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException;
  4. //bean初始化方法调用后被调用
  5. Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException;
  6. }

如上接口声明所示,BeanPostProcessor接口有两个回调方法。当一个BeanPostProcessor的实现类注册到Spring IOC容器后,对于该Spring IOC容器所创建的每个bean实例在初始化方法(如afterPropertiesSet和任意已声明的init方法)调用前,将会调用BeanPostProcessor中的postProcessBeforeInitialization方法,而在bean实例初始化方法调用完成后,则会调用BeanPostProcessor中的postProcessAfterInitialization方法,整个调用顺序可以简单示意如下:

--> Spring IOC容器实例化Bean
--> 调用BeanPostProcessor的postProcessBeforeInitialization方法
--> 调用bean实例的初始化方法
--> 调用BeanPostProcessor的postProcessAfterInitialization方法

可以看到,Spring容器通过BeanPostProcessor给了我们一个机会对Spring管理的bean进行再加工。比如:我们可以修改bean的属性,可以给bean生成一个动态代理实例等等。一些Spring AOP的底层处理也是通过实现BeanPostProcessor来执行代理包装逻辑的。

BeanPostProcessor实战

了解了BeanPostProcessor的相关知识后,下面我们来通过项目中的一个具体例子来体验一下它的神奇功效吧。

先介绍一下我们的项目背景吧:我们项目中经常会涉及AB 测试,这就会遇到同一套接口会存在两种不同实现。实验版本与对照版本需要在运行时同时存在。下面用一些简单的类来做一个示意:

  1. public class HelloService{
  2. void sayHello();
  3. void sayHi();
  4. }

HelloService有以下两个版本的实现:

  1. @Service
  2. public class HelloServiceImplV1 implements HelloService{
  3. public void sayHello(){
  4. System.out.println("Hello from V1");
  5. }
  6. public void sayHi(){
  7. System.out.println("Hi from V1");
  8. }
  9. }
  1. @Service
  2. public class HelloServiceImplV2 implements HelloService{
  3. public void sayHello(){
  4. System.out.println("Hello from V2");
  5. }
  6. public void sayHi(){
  7. System.out.println("Hi from V2");
  8. }
  9. }

做AB测试的话,在使用BeanPostProcessor封装前,我们的调用代码大概是像下面这样子的:

  1. @Controller
  2. public class HelloController{
  3. @Autowird
  4. private HelloServiceImplV1 helloServiceImplV1
  5. @Autowird
  6. private HelloServiceImplV2 helloServiceImplV2
  7. public void sayHello(){
  8. if(getHelloVersion()=="A"){
  9. helloServiceImplV1.sayHello();
  10. }else{
  11. helloServiceImplV2.sayHello();
  12. }
  13. }
  14. public void sayHi(){
  15. if(getHiVersion()=="A"){
  16. helloServiceImplV1.sayHi();
  17. }else{
  18. helloServiceImplV2.sayHi();
  19. }
  20. }
  21. }

可以看到,这样的代码看起来十分不优雅,并且如果AB测试的功能点很多的话,那项目中就会充斥着大量的这种重复性分支判断,看到代码就想死有木有!!!维护代码也将会是个噩梦。比如某个功能点AB测试完毕,需要把全部功能切换到V2版本,V1版本不再需要维护,那么处理方式有两种:

  • 把A版本代码留着不管:这将会导致到处都是垃圾代码从而造成代码臃肿难以维护
  • 找到所有V1版本被调用的地方然后把相关分支删掉:这很容易在处理代码的时候删错代码从而造成生产事故。

怎么解决这个问题呢,我们先看代码,后文再给出解释:

  1. @Target({ElementType.FIELD})
  2. @Retention(RetentionPolicy.RUNTIME)
  3. @Documented
  4. @Component
  5. public @interface RoutingInjected{
  6. }
  1. @Target({ElementType.FIELD,ElementType.METHOD})
  2. @Retention(RetentionPolicy.RUNTIME)
  3. @Documented
  4. @Component
  5. public @interface RoutingSwitch{
  6. /**
  7. * 在配置系统中开关的属性名称,应用系统将会实时读取配置系统中对应开关的值来决定是调用哪个版本
  8. * @return
  9. */
  10. String value() default "";
  11. }

  1. @RoutingSwitch("hello.switch")
  2. public class HelloService{
  3. @RoutingSwitch("A")
  4. void sayHello();
  5. void sayHi();
  6. }
  1. @Controller
  2. public class HelloController{
  3. @RoutingInjected
  4. private HelloService helloService
  5. public void sayHello(){
  6. this.helloService.sayHello();
  7. }
  8. public void sayHi(){
  9. this.helloService.sayHi();
  10. }
  11. }

现在我们可以停下来对比一下封装前后调用代码了,是不是感觉改造后的代码优雅很多呢?那么这是怎么实现的呢,我们一起来揭开它的神秘面纱吧,请看代码:

  1. @Component
  2. public class RoutingBeanPostProcessor implements BeanPostProcessor {
  3. @Autowired
  4. private ApplicationContext applicationContext;
  5. @Override
  6. public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
  7. return bean;
  8. }
  9. @Override
  10. public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
  11. Class clazz = bean.getClass();
  12. Field[] fields = clazz.getDeclaredFields();
  13. for (Field f : fields) {
  14. if (f.isAnnotationPresent(RoutingInjected.class)) {
  15. if (!f.getType().isInterface()) {
  16. throw new BeanCreationException("RoutingInjected field must be declared as an interface:" + f.getName()
  17. + " @Class " + clazz.getName());
  18. }
  19. try {
  20. this.handleRoutingInjected(f, bean, f.getType());
  21. } catch (IllegalAccessException e) {
  22. throw new BeanCreationException("Exception thrown when handleAutowiredRouting", e);
  23. }
  24. }
  25. }
  26. return bean;
  27. }
  28. private void handleRoutingInjected(Field field, Object bean, Class type) throws IllegalAccessException {
  29. Map<String, Object> candidates = this.applicationContext.getBeansOfType(type);
  30. field.setAccessible(true);
  31. if (candidates.size() == 1) {
  32. field.set(bean, candidates.values().iterator().next());
  33. } else if (candidates.size() == 2) {
  34. Object proxy = RoutingBeanProxyFactory.createProxy(type, candidates);
  35. field.set(bean, proxy);
  36. } else {
  37. throw new IllegalArgumentException("Find more than 2 beans for type: " + type);
  38. }
  39. }
  40. }
  1. public class RoutingBeanProxyFactory {
  2. public static Object createProxy(Class targetClass, Map<String, Object> beans) {
  3. ProxyFactory proxyFactory = new ProxyFactory();
  4. proxyFactory.setInterfaces(targetClass);
  5. proxyFactory.addAdvice(new VersionRoutingMethodInterceptor(targetClass, beans));
  6. return proxyFactory.getProxy();
  7. }
  8. static class VersionRoutingMethodInterceptor implements MethodInterceptor {
  9. private String classSwitch;
  10. private Object beanOfSwitchOn;
  11. private Object beanOfSwitchOff;
  12. public VersionRoutingMethodInterceptor(Class targetClass, Map<String, Object> beans) {
  13. String interfaceName = StringUtils.uncapitalize(targetClass.getSimpleName());
  14. if(targetClass.isAnnotationPresent(RoutingSwitch.class)){
  15. this.classSwitch =((RoutingSwitch)targetClass.getAnnotation(RoutingSwitch.class)).value();
  16. }
  17. this.beanOfSwitchOn = beans.get(this.buildBeanName(interfaceName, true));
  18. this.beanOfSwitchOff = beans.get(this.buildBeanName(interfaceName, false));
  19. }
  20. private String buildBeanName(String interfaceName, boolean isSwitchOn) {
  21. return interfaceName + "Impl" + (isSwitchOn ? "V2" : "V1");
  22. }
  23. @Override
  24. public Object invoke(MethodInvocation invocation) throws Throwable {
  25. Method method = invocation.getMethod();
  26. String switchName = this.classSwitch;
  27. if (method.isAnnotationPresent(RoutingSwitch.class)) {
  28. switchName = method.getAnnotation(RoutingSwitch.class).value();
  29. }
  30. if (StringUtils.isBlank(switchName)) {
  31. throw new IllegalStateException("RoutingSwitch's value is blank, method:" + method.getName());
  32. }
  33. return invocation.getMethod().invoke(getTargetBean(switchName), invocation.getArguments());
  34. }
  35. public Object getTargetBean(String switchName) {
  36. boolean switchOn;
  37. if (RoutingVersion.A.equals(switchName)) {
  38. switchOn = false;
  39. } else if (RoutingVersion.B.equals(switchName)) {
  40. switchOn = true;
  41. } else {
  42. switchOn = FunctionSwitch.isSwitchOpened(switchName);
  43. }
  44. return switchOn ? beanOfSwitchOn : beanOfSwitchOff;
  45. }
  46. }
  47. }

我简要解释一下思路:

  • 首先自定义了两个注解:RoutingInjected、RoutingSwitch,前者的作用类似于我们常用的Autowired,声明了该注解的属性将会被注入一个路由代理类实例;后者的作用则是一个配置开关,声明了控制路由的开关属性
  • 在RoutingBeanPostProcessor类中,我们在postProcessAfterInitialization方法中通过检查bean中是否存在声明了RoutingInjected注解的属性,如果发现存在该注解则给该属性注入一个动态代理类实例
  • RoutingBeanProxyFactory类功能就是生成一个代理类实例,代理类的逻辑也比较简单。版本路由支持到方法级别,即优先检查方法是否存在路由配置RoutingSwitch,方法不存在配置时才默认使用类路由配置

作者:圆圆仙人球
链接:https://www.jianshu.com/p/1417eefd2ab1
来源:简书
简书著作权归作者所有,任何形式的转载都请联系作者获得授权并注明出处。

BeanPostProcessor(转)的更多相关文章

  1. spring源码:BeanPostProcessor(li)

    在spring管理Bean的初始化过程中,除了正常管理bean的实例化(初始化.参数注入等)外,还对外提供了丰富的对Bean操作的扩展.例如自定义初始化操作,自定义容器退出时Bean的销毁操作等等.这 ...

  2. [spring源码学习]五-BeanPostProcessor的使用

    一.接口描述 spring提供了一个接口类-BeanPostProcessor,我们叫他:bean的加工器,应该是在bean的实例化过程中对bean做一些包装处理,里边提供两个方法 public in ...

  3. spring 源码分析之BeanPostProcessor

    1.官方解答: Factory hook that allows for custom modification of new bean instances, e.g. checking for ma ...

  4. 04Spring_bean 后处理器(后处理Bean),BeanPostProcessor ,bean创建时序,动态代理

    这篇文章很重要,讲解的是动态代理,以及bean创建前后的所发生的事情.介绍一个接口:在Spring构造Bean对象过程中,有一个环节对Bean对象进行 后处理操作 (钩子函数) ----- Sprin ...

  5. spring 后置处理器BeanFactoryPostProcessor和BeanPostProcessor的用法和区别

    主要区别就是: BeanFactoryPostProcessor可以修改BEAN的配置信息而BeanPostProcessor不能,下面举个例子说明 BEAN类: package com.spring ...

  6. 【Spring】对象后期处理,BeanPostProcessor

    当我们使用Spring容器管理对象时,需要对对象进行一些后期处理时,比如数据处理.数据预加载,可以使用BeanPostProcessor接口. 简单演示它的用法. 定义扫描包,显示定义BeanPost ...

  7. Spring的BeanFactoryPostProcessor和BeanPostProcessor

    转载:http://blog.csdn.net/caihaijiang/article/details/35552859 BeanFactoryPostProcessor和BeanPostProces ...

  8. Spring 的 BeanPostProcessor接口实现

    今天学习了一下Spring的BeanPostProcessor接口,该接口作用是:如果我们需要在Spring容器完成Bean的实例化,配置和其他的初始化后添加一些自己的逻辑处理,我们就可以定义一个或者 ...

  9. Spring中BeanPostProcessor

    Spring中BeanPostProcessor 前言: 本文旨在介绍Spring动态配置数据源的方式,即对一个DataSource的配置诸如jdbcUrl,user,password,driverC ...

  10. BeanPostProcessor 的使用,实现在对象初始化之前或者之后对对象进行操作

    import java.lang.reflect.Field; import org.springframework.beans.BeansException; import org.springfr ...

随机推荐

  1. Function Expression

    One of the key characteristics of function declarations is function declaration hoisting, whereby fu ...

  2. windows vs2015 编译openssl 1.1.0c

    1,到openssl官网下载源码. 2,安装activePerl,我放在网盘:https://pan.baidu.com/s/1ZHe24yRcPtIuSiEa-3oqxw 3.安装完毕后,使用 VS ...

  3. video标签在移动端的一些属性值设置

    <video x5-video-orientation="portraint" src="" loop x-webkit-airplay="al ...

  4. Appium - multiprocessing.pool.MaybeEncodingError-【 “Can’t pickle local object ‘PoolManager.__init__.<locals>.<lambda>‘】

    公司同事学习自动化新装环境后,run多进程测试用例时出错: multiprocessing.pool.MaybeEncodingError: Error sending result: ’<ap ...

  5. java 换包

    1.进入目录:cd /opt/cw_isc/ 2.查看:ls 3.查看进程:ps -ef|grep aw-sc-package-all.jar 4.杀死进程kill -9 3982 5.查看进程:ps ...

  6. Qt - 基于TCP的网络编程

    TCP(传输控制协议 Transmission Control Protocol) 可靠.面向数据流.面向连接  的传输协议.(许多应用层协议都是以它为基础:HTTP.FTP) 使用情况: 相比UDP ...

  7. Virtual DOM和snabbdom.js

    Virtual DOM和snabbdom.js:https://www.jianshu.com/p/1f1ef915e83e

  8. Servlet概念及与Jsp的区别

    一.Servlet概念 Servlet是在服务器上运行的小程序.一个Servlet就是一个Java类,并且可以通过”请求-响应”编程模型来访问这个驻留在服务器内存里的Servlet程序 二.Servl ...

  9. Druid + spring 配置数据库连接池

    1. Druid的简介 Druid是一个数据库连接池.Druid是目前最好的数据库连接池,在功能.性能.扩展性方面,都超过其他数据库连接池,包括DBCP.C3P0.BoneCP.Proxool.JBo ...

  10. [NodeJs系列]聊一聊BOM

    最近在看Node源码的时候,偶然间,看到如下函数: /** * Remove byte order marker. This catches EF BB BF (the UTF-8 BOM) * be ...