覆盖重写 原有Spring Bean的几种方式
目录
- 场景
- 方法1 直接在自己工程中建同包同类名的类进行替换
- 方法2 采用@Primary注解
- 方法3 排除需要替换的jar包中的类
- 方法4 @Bean 覆盖
- 方法5 使用BeanDefinitionRegistryPostProcessor
场景
什么情况下要覆写原有的Spring Bean ? 例如引入的第三方jar包中的某个类有些问题,然有没有源码提供或者嫌编译源码太费事,这个时间可以考虑覆写原有的类。
方法1 直接在自己工程中建同包同类名的类进行替换
方式简单粗暴,可以直接覆盖掉jar包中的类,spring项目会优先加载自定义的类。
下面是覆盖 flowable框架中的一个类 FlowableCookieFilter,主要是想修改它里面的redirectToLogin方法的业务逻辑。包路径为 org.flowable.ui.common.filter, 直接工程里面新建一样路径一样类名FlowableCookieFilter即可。
方法2 采用@Primary注解
该方法适用于接口实现类,自己创建一个原jar包接口的实现类,然后类上加上@Primary注解,spring则默认加载该类实例化出的Bean。
下面的例子: 一个接口 RemoteIdmService,原先jar包中只有一个实现类 RemoteIdmServiceImpl,现在自己工程里面创建一个 CustomRemoteIdmServiceImpl 继承RemoteIdmService接口,然后发现所有调用RemoteIdmService接口里面的方法实际调用走的是CustomRemoteIdmServiceImpl 里面的方法。


方法3 排除需要替换的jar包中的类
使用 @ComponentScan 里面的 excludeFilters 功能排除调用要替换的类,然后自己写个类继承替换的类即可。
下面的例子是替换掉 jar包中的PersistentTokenServiceImpl类
@SpringBootApplication@ComponentScan(excludeFilters = {@ComponentScan.Filter(type =FilterType.ASSIGNABLE_TYPE, classes = {PersistentTokenServiceImpl.class})})public class Application {public static void main(String[] args) {new SpringApplication(Test.class).run(args);}}
@Servicepublic class MyPersistentTokenServiceImpl extends PersistentTokenServiceImpl{@Overridepublic Token saveAndFlush(Token token) {// 覆写该方法的业务逻辑tokenCache.put(token.getId(), token);idmIdentityService.saveToken(token);return token;}@Overridepublic void delete(Token token) {// 覆写该方法的业务逻辑tokenCache.invalidate(token.getId());idmIdentityService.deleteToken(token.getId());}@Overridepublic Token getPersistentToken(String tokenId) {// 覆写该方法的业务逻辑return getPersistentToken(tokenId, false);}}
方法4 @Bean 覆盖
该场景针对,框架jar包中有@ConditionalOnMissingBean注解,这种注解是说明如果你也创建了一个一样的Bean则框架就不自己再次创建这个bean了,这样你可以覆写自己的bean。
原jar包中的配置类:
直接继承要覆盖的类,自己重写里面方法,使用@Component注入到spring中去
方法5 使用BeanDefinitionRegistryPostProcessor
关于 BeanDefinitionRegistryPostProcessor 、 BeanFactoryPostProcessor可以参考下面的博文:
https://blog.csdn.net/ztchun/article/details/90814135
BeanDefinitionRegistryPostProcessor 说白了就是可以在初始化Bean之前修改Bean的属性,甚至替换原先准备要实例化的bean。
实战演示:
假设jar包中有一个类 MyTestService,正常情况下它会被spring自动扫描到注入IOC容器中去。
package com.middol.mytest.config.beantest.register.jar;import org.springframework.stereotype.Service;import javax.annotation.PostConstruct;import javax.annotation.PreDestroy;/** * @author guzt */@Service("myTestService")public class MyTestService {private String name1;private String name2;private String name3;public MyTestService() {this.name1 = "";this.name2 = "";this.name3 = "";}public MyTestService(String name1, String name2, String name3) {this.name1 = name1;this.name2 = name2;this.name3 = name3;}@PostConstructpublic void init() {System.out.println("MyTestService init");}@PreDestroypublic void destory() {System.out.println("MyTestService destroy");}public void show() {System.out.println("------------------------");System.out.println("我是jar中通过注解@Service主动加入Spring的IOC里面的");System.out.println("------------------------");}public String getName1() {return name1;}public void setName1(String name1) {this.name1 = name1;}public String getName2() {return name2;}public void setName2(String name2) {this.name2 = name2;}public String getName3() {return name3;}public void setName3(String name3) {this.name3 = name3;}}
自己工程中继承该类,并且覆写里面的show中的方法
package com.middol.mytest.config.beantest.register;import com.middol.mytest.config.beantest.register.jar.MyTestService;/** * @author guzt */public class MyTestServiceIpml extends MyTestService {public MyTestServiceIpml() {}public MyTestServiceIpml(String name1, String name2, String name3) {super(name1, name2, name3);}@Overridepublic void show() {System.out.println("------------------------");System.out.println("我是被BeanDefinitionRegistry手动注册到Spring的IOC里面的");System.out.println("------------------------");}}
然后 实现 BeanDefinitionRegistryPostProcessor 接口,修改原来bean定义,主要查看postProcessBeanDefinitionRegistry方法的实现,先清空原bean定义,注册我们自己的bean定义来达到替换的目的。
package com.middol.mytest.config.beantest.register;import org.slf4j.Logger;import org.slf4j.LoggerFactory;import org.springframework.beans.BeansException;import org.springframework.beans.factory.config.ConfigurableListableBeanFactory;import org.springframework.beans.factory.support.BeanDefinitionBuilder;import org.springframework.beans.factory.support.BeanDefinitionRegistry;import org.springframework.beans.factory.support.BeanDefinitionRegistryPostProcessor;import org.springframework.stereotype.Component;import org.springframework.web.bind.annotation.RestController;import java.util.Map;/** * @author amdin */@Componentpublic class MyBeanDefinitionRegistryPostProcessor implements BeanDefinitionRegistryPostProcessor {private Logger logger = LoggerFactory.getLogger(this.getClass());@Overridepublic void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry beanDefinitionRegistry) throws BeansException {logger.info("bean 定义查看和修改...");String beanName = "myTestService";// 先移除原来的bean定义beanDefinitionRegistry.removeBeanDefinition(beanName);// 注册我们自己的bean定义BeanDefinitionBuilder beanDefinitionBuilder = BeanDefinitionBuilder.rootBeanDefinition(MyTestServiceIpml.class);// 如果有构造函数参数, 有几个构造函数的参数就设置几个 没有就不用设置beanDefinitionBuilder.addConstructorArgValue("构造参数1");beanDefinitionBuilder.addConstructorArgValue("构造参数2");beanDefinitionBuilder.addConstructorArgValue("构造参数3");// 设置 init方法 没有就不用设置beanDefinitionBuilder.setInitMethodName("init");// 设置 destory方法 没有就不用设置beanDefinitionBuilder.setDestroyMethodName("destory");// 将Bean 的定义注册到Spring环境beanDefinitionRegistry.registerBeanDefinition("myTestService", beanDefinitionBuilder.getBeanDefinition());}@Overridepublic void postProcessBeanFactory(ConfigurableListableBeanFactory configurableListableBeanFactory) throws BeansException {// bean的名字为key, bean的实例为valueMap<String, Object> beanMap = configurableListableBeanFactory.getBeansWithAnnotation(RestController.class);logger.info("所有 RestController 的bean {}", beanMap);}}
写一个 业务类BusinessTestService测试一下,期望结果:所有用到 MyTestService的地方实际调用的变成了MyTestServiceIpml里面的方法。
package com.middol.mytest.config.beantest.register;import com.middol.mytest.config.beantest.register.jar.MyTestService;import org.springframework.stereotype.Service;import javax.annotation.PostConstruct;import javax.annotation.Resource;/** * @author guzt */@Servicepublic class BusinessTestService {@Resourceprivate MyTestService myTestService;@PostConstructpublic void init() {System.out.println(myTestService.getName1());System.out.println(myTestService.getName2());System.out.println(myTestService.getName3());// 看看到底是哪一个BeanmyTestService.show();}}
控制台打印如下:
可以发现,和我们期望的结果的一样:所有用到 MyTestService的地方实际调用的变成了MyTestServiceIpml里面的方法 !
OVER …
转自:https://www.wangt.cc/2020/10/%E8%A6%86%E7%9B%96%E9%87%8D%E5%86%99-%E5%8E%9F%E6%9C%89spring-bean%E7%9A%84%E5%87%A0%E7%A7%8D%E6%96%B9%E5%BC%8F/
浙公网安备 33010602011771号