位置: IT常识 - 正文

【Spring Boot】SpringBoot设计了哪些可拓展的机制?(spring boot s)

编辑:rootadmin
【Spring Boot】SpringBoot设计了哪些可拓展的机制? 文章目录前言SpringBoot核心源码拓展Initializer拓展监听器ApplicationListenerBeanFactory的后置处理器 & Bean的后置处理器AOP其他的拓展点前言 当我们引入注册中心的依赖,比如nacos的时候,当我们启动springboot,这个服务就会根据配置文件自动注册到注册中心中,这个动作是如何完成的? 注册中心使用了SpringBoot中的事件监听机制,在springboot初始化的时候完成服务注册SpringBoot核心源码public SpringApplication(ResourceLoader resourceLoader, Class<?>... primarySources) { ... this.primarySources = new LinkedHashSet(Arrays.asList(primarySources)); // Servlet this.webApplicationType = WebApplicationType.deduceFromClasspath(); this.bootstrapRegistryInitializers = new ArrayList(this.getSpringFactoriesInstances(BootstrapRegistryInitializer.class)); // 注意这里,Initializers this.setInitializers(this.getSpringFactoriesInstances(ApplicationContextInitializer.class)); // 注意这里 Listeners this.setListeners(this.getSpringFactoriesInstances(ApplicationListener.class)); this.mainApplicationClass = this.deduceMainApplicationClass(); }

推荐整理分享【Spring Boot】SpringBoot设计了哪些可拓展的机制?(spring boot s),希望有所帮助,仅作参考,欢迎阅读内容。

文章相关热门搜索词:spring boot poi,spring boot 揭秘,spring boot s,spring spring boot,spring-boot-actuator,spring boot+ssm,spring spring boot,spring boot spi,内容如对您有帮助,希望把文章链接给更多的朋友!

我们可以看到空的SpringBoot项目有一些initializers以及一些listeners

注意这两行,换言之我们只要实现这两个类就可以自定义拓展SpringBoot了!

这里和手写Starter都是对SpringBoot的拓展,有兴趣的小伙伴可以看这篇文章

拓展Initializer

再看这张图

我们需要研究一下ApplicationContextInitializer这个类:

@FunctionalInterfacepublic interface ApplicationContextInitializer<C extends ConfigurableApplicationContext> { /** * Initialize the given application context. * @param applicationContext the application to configure */ void initialize(C applicationContext); }

这样就很清晰了,我们尝试手写一个继承类:

public class DemoInitializer implements ApplicationContextInitializer { @Override public void initialize(ConfigurableApplicationContext applicationContext) { System.out.println("自定义初始化器执行..."); ConfigurableEnvironment environment = applicationContext.getEnvironment(); Map<String, Object> map = new HashMap<>(1); map.put("name", "sccccc"); environment.getPropertySources().addLast(new MapPropertySource("DemoInitializer", map)); System.out.println("DemoInitializer execute, and add some property"); } }

通过SPI机制将自定义初始化器交给list集合initializers

然后再debug,就会发现:

【Spring Boot】SpringBoot设计了哪些可拓展的机制?(spring boot s)

最后经过一次回调:

private void prepareContext(DefaultBootstrapContext bootstrapContext, ConfigurableApplicationContext context, ... applyInitializers(context); ... // Add boot specific singleton beans 下面是beanFactory的操作

遍历所有的初始化器,然后

/*** Apply any {@link ApplicationContextInitializer}s to the context before it is * refreshed. * @param context the configured ApplicationContext (not refreshed yet) * @see ConfigurableApplicationContext#refresh() */ @SuppressWarnings({ "rawtypes", "unchecked" }) protected void applyInitializers(ConfigurableApplicationContext context) { for (ApplicationContextInitializer initializer : getInitializers()) { Class<?> requiredType = GenericTypeResolver.resolveTypeArgument(initializer.getClass(), ApplicationContextInitializer.class); Assert.isInstanceOf(requiredType, context, "Unable to call initializer."); initializer.initialize(context); } }

流程:

拓展监听器ApplicationListener

@FunctionalInterfacepublic interface ApplicationListener<E extends ApplicationEvent> extends EventListener { /** * Handle an application event. */ void onApplicationEvent(E event); /** * Create a new {@code ApplicationListener} for the given payload consumer. */ static <T> ApplicationListener<PayloadApplicationEvent<T>> forPayload(Consumer<T> consumer) { return event -> consumer.accept(event.getPayload()); } }

这里和上面initializer一样,就不演示了

BeanFactory的后置处理器 & Bean的后置处理器

Spring Boot解析配置成BeanDefinition的操作在invokeBeanFactoryPostProcessors方法中 自定义BeanFactory的后置处理器:

@Componentpublic class MyBeanFactoryPostProcessor implements BeanFactoryPostProcessor { @Override public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException { Arrays.asList(beanFactory.getBeanDefinitionNames()) .forEach(beanDefinitionName -> System.out.println(beanDefinitionName)); System.out.println("BeanFactoryPostProcessor..."); }}

自定义Bean的后置处理器:

@Componentpublic class MyBeanPostProcessor implements BeanPostProcessor { @Override public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException { if(beanName.equals("userController")){ System.out.println("找到了userController: "+bean); } return null; }}AOP

这个相信大家用的比较多,可以自定义切面:

@Aspect@Componentpublic class LogAspect {// 切入点 Pointcut 可以对Service服务做切面@Pointcut("execution(* com.example.service.*.*(..))")public void mypointcut(){}// 前置通知@Before(value = "mypointcut()")public void before(JoinPoint joinPoint){ System.out.println("[前置通知] 准备开始记录日志..."); System.out.println("[前置通知] 目标类是: "+joinPoint.getTarget()); System.out.println("[前置通知] 目标方法是: "+joinPoint.getSignature().getName());}// 后置通知@AfterReturning(value = "mypointcut()")public void afterReturning(JoinPoint joinPoint){ System.out.println("[后置通知] 记录日志完成..."); System.out.println("[后置通知] 目标类是: "+joinPoint.getTarget()); System.out.println("[后置通知] 目标方法是: "+joinPoint.getSignature().getName());}/*@Around(value = "mypointcut()")public void around(ProceedingJoinPoint joinPoint){ System.out.println("[环绕通知] 日志记录前的操作..."); try { joinPoint.proceed(); System.out.println("[环绕通知] 日志记录后的操作..."); System.out.println("[环绕通知] "+joinPoint.getTarget()); System.out.println("[环绕通知] "+joinPoint.getSignature().getName()); } catch (Throwable throwable) { System.out.println("[环绕通知] 发生异常的操作..."); throwable.printStackTrace(); }finally { ... }}其他的拓展点Banner

方法地址: printBanner(env)->bannerPrinter.print->SpringBootBanner#printBanner 可以在resource目录下建立banner.txt文件夹实现自定义Banner

Runners

流程:

自定义:

@Componentpublic class JackApplicationRunner implements ApplicationRunner { @Override public void run(ApplicationArguments args) throws Exception { System.out.println("JackApplicationRunner..."); }}
本文链接地址:https://www.jiuchutong.com/zhishi/300352.html 转载请保留说明!

上一篇:OpenAI GPT-3模型详解(gpt3 模型大小)

下一篇:浅谈DNS域名解析的过程(dns域名解析两种方式)

  • 增值税发票选择确认平台打不开怎么办?
  • 烟叶税计入什么
  • 一般纳税人购进货物用于免征增值税项目
  • 进项税和销项税税率一样吗
  • 其他个人出租房屋按照多少征收应纳税额
  • 借条时间到了怎么续
  • 未确认融资收益怎么算
  • 企业管理费可以扣除吗
  • 母公司对子公司的债务承担连带责任
  • 直系亲属之间转账要交税吗
  • 资产减值损失确认后,减值资产的折旧
  • 公司清算关闭程序
  • 出租屋物品损坏怎么赔偿
  • 土地租赁费含有多少税率
  • 小型微利企业所得税优惠政策2022
  • 资本公积转增资本个人股东是否纳税
  • 购入商品怎么做会计分录
  • 初级考试备考计划
  • 发票未到计提房租费用会计分录
  • 表格怎样打印在一张a4纸上
  • 苹果的mac系统
  • 使用的磁盘空间在哪
  • 虚拟机安装apache
  • 笔记本如何打开无线网络开关
  • win10怎么打开移动中心
  • php中变量有哪些基本数据类型?
  • hbuilderx怎么运行代码
  • PHP aes (ecb)解密后乱码问题
  • file php
  • 生产企业免抵退税实例
  • 企业经营状态有哪几种
  • 北极光的征兆
  • 普通发票和增值税发票都能报销吗
  • st的电机库性能怎么样呢
  • vue3项目实战
  • 微信小程序开发者工具
  • Code For Better 谷歌开发者之声——使用谷歌浏览器 Chrome 更好地调试
  • 工业企业采购流程
  • phpcms教程
  • 文化事业建设费税收优惠政策2023
  • 手机银行对公账户转账要手续费吗
  • Sqlserver 2005使用XML一次更新多条记录的方法
  • 工程项目甲方是什么意思
  • 企业支付的佣金计算多少税率呢
  • 如何核算购进商品
  • 长期借款计提的利息可能借记
  • 收到银行手续费发票怎么入账
  • 企业支付宝扣费
  • 公司账户收入多少要扣税
  • 预计负债可以获取利润吗
  • 其他应付款的有
  • 销售方开具的红字专票购买方在税控盘要怎么操作
  • 资产负债表期初余额和期末余额
  • 小规模公司月末报税吗
  • 出口退税怎么回事
  • 预收账款余额在贷方是什么意思
  • 作废发票丢失怎么补办
  • sql2008设置单用户模式
  • sql server 复制数据库具体操作图解
  • win7支持net.framework4.7.2
  • win10怎么用ghost
  • windowsserver2008密码规则
  • linux 软件 安装
  • 电脑卡 硬盘100%
  • ubuntu独立显卡黑屏
  • linux的grep命令详解
  • win8如何进行系统还原
  • lsass.exe是什么病毒
  • Win7中TrustedInstaller.exe进程占用内存高该怎么解决?
  • win7激活2020
  • Win7系统关闭自动更新
  • appiumforwindows的简单安装和启动(安卓)
  • 创建步骤
  • 微信内置浏览器怎么清除缓存
  • 魔兽世界全部开发成本
  • noodoe如何使用
  • JavaScript中定义数组
  • 出口退税网上申报流程
  • 哪些账本需要贴纸
  • 转卖车辆车船税多少
  • 免责声明:网站部分图片文字素材来源于网络,如有侵权,请及时告知,我们会第一时间删除,谢谢! 邮箱:opceo@qq.com

    鄂ICP备2023003026号

    网站地图: 企业信息 工商信息 财税知识 网络常识 编程技术

    友情链接: 武汉网站建设