首页 文章 精选 留言 我的

精选列表

搜索[网站开发],共10000篇文章
优秀的个人博客,低调大师

Spring注解驱动开发之七——BeanPostProcessor 执行原理、在spring 中的应用

本文包含以下内容: BeanPostProcessor 执行原理 断点查看 总结运行过程中的执行原理 BeanPostProcessor 在spring 中的应用 ApplicationContextAwareProcessor帮助组件注入IOC 容器 InitDestroyAnnotationBeanPostProcessor用于处理 @PostConstruct 、@PreDestroy 注解 BeanValidationPostProcessor 数据校验 1.BeanPostProcessor 执行原理 1)断点查看代码 1.在MyBeanPostProcessor添加断点,运行测试方法通过观察栈,查看执行流程 下方长长的英文是调试中右下角的窗口: 2. <init>:84, AnnotationConfigApplicationContext 初始化容器 public AnnotationConfigApplicationContext(Class<?>... annotatedClasses) { this(); register(annotatedClasses); refresh(); } 3. refresh:543, AbstractApplicationContext 调用refresh()函数 ,实例化所有剩余的单例 // Instantiate all remaining (non-lazy-init) singletons. finishBeanFactoryInitialization(beanFactory); 4 .finishBeanFactoryInitialization:867, AbstractApplicationContext // Instantiate all remaining (non-lazy-init) singletons. beanFactory.preInstantiateSingletons(); preInstantiateSingletons:761, DefaultListableBeanFactory,具体通过getBean函数进行初始化剩余的Bean getBean(beanName); 5. doGetBean:302, AbstractBeanFactory 这一步 通过getSingleton 获取单实例, 通过初始化createBean 方法创建Bean // Create bean instance. if (mbd.isSingleton()) { sharedInstance = getSingleton(beanName, new ObjectFactory<Object>() { @Override public Object getObject() throws BeansException { try { return createBean(beanName, mbd, args); } catch (BeansException ex) { // Explicitly remove instance from singleton cache: It might have been put there // eagerly by the creation process, to allow for circular reference resolution. // Also remove any beans that received a temporary reference to the bean. destroySingleton(beanName); throw ex; } } }); bean = getObjectForBeanInstance(sharedInstance, name, beanName, mbd); } 6. createBean:483, AbstractAutowireCapableBeanFactory,调用初始化,获取Bean 实例 Object beanInstance = doCreateBean(beanName, mbdToUse, args); 7. doCreateBean:555, AbstractAutowireCapableBeanFactory 进入到具体创建Bean 中 通过populateBean() 进行填充Bean 操作 Object exposedObject = bean; try { populateBean(beanName, mbd, instanceWrapper); if (exposedObject != null) { exposedObject = initializeBean(beanName, exposedObject, mbd); } } 8.最后可以看到 initializeBean:1620, AbstractAutowireCapableBeanFactory 中执行 applyBeanPostProcessorsBeforeInitialization和 applyBeanPostProcessorsAfterInitialization进行 postProcessBeforeInitialization()函数和postProcessAfterInitialization()函数的调用;中间调用 invokeInitMethods(beanName, wrappedBean, mbd) ;进行 初始化 函数的调用 Object wrappedBean = bean; if (mbd == null || !mbd.isSynthetic()) { wrappedBean = applyBeanPostProcessorsBeforeInitialization(wrappedBean, beanName); } try { invokeInitMethods(beanName, wrappedBean, mbd); } catch (Throwable ex) { throw new BeanCreationException( (mbd != null ? mbd.getResourceDescription() : null), beanName, "Invocation of init method failed", ex); } if (mbd == null || !mbd.isSynthetic()) { wrappedBean = applyBeanPostProcessorsAfterInitialization(wrappedBean, beanName); } 9. applyBeanPostProcessorsBeforeInitialization:409,AbstractAutowireCapableBeanFactory 最后通过 getBeanPostProcessors() 扫描所有的 BeanPostProcessor 直到遇到null 进行返回退出循环 @Override public Object applyBeanPostProcessorsBeforeInitialization(Object existingBean, String beanName) throws BeansException { Object result = existingBean; for (BeanPostProcessor beanProcessor : getBeanPostProcessors()) { result = beanProcessor.postProcessBeforeInitialization(result, beanName); if (result == null) { return result; } } return result; } 2) 总结 运行过程中的执行原理 通过查看断点调试结果, 可以看到,创建单例容器Bean 通过 populateBean() 函数进行填充,invokeInitMethods(beanName, wrappedBean, mbd); 函数进行初始化,然后前后有applyBeanPostProcessorsBeforeInitialization和applyBeanPostProcessorsAfterInitialization 进行对所有的BeanPostProcessor 扫描执行。 2.BeanPostProcessor 在spring 中的应用 在spring 中大量使用BeanPostProcessor ,进行处理,下面举例3个进行断点、代码测试: 1).ApplicationContextAwareProcessor 帮助组件注入IOC 容器 在Dog 中实现 implements ApplicationContextAware 接口,编写, setApplicationContext 将容器保存至属性方法中 @ComponentpublicclassDogimplementsApplicationContextAware{ private ApplicationContext applicationContext; public Dog(){ System.out.println("dog constructor..."); } @Override public void setApplicationContext(ApplicationContext applicationContext) throws BeansException { // TODO Auto-generated method stub this.applicationContext = applicationContext; }} 1. postProcessBeforeInitialization:97,ApplicationContextAwareProcessor 可以看到 通过 postProcessBeforeInitialization 函数进行将容器进行赋值 2.通过invokeAwareInterfaces判断Bean 的类型进行类型转换,并进行赋值 private void invokeAwareInterfaces(Object bean) { if (bean instanceof Aware) { if (bean instanceof EnvironmentAware) { ((EnvironmentAware) bean).setEnvironment(this.applicationContext.getEnvironment()); } if (bean instanceof EmbeddedValueResolverAware) { ((EmbeddedValueResolverAware) bean).setEmbeddedValueResolver(this.embeddedValueResolver); } if (bean instanceof ResourceLoaderAware) { ((ResourceLoaderAware) bean).setResourceLoader(this.applicationContext); } if (bean instanceof ApplicationEventPublisherAware) { ((ApplicationEventPublisherAware) bean).setApplicationEventPublisher(this.applicationContext); } if (bean instanceof MessageSourceAware) { ((MessageSourceAware) bean).setMessageSource(this.applicationContext); } if (bean instanceof ApplicationContextAware) { ((ApplicationContextAware) bean).setApplicationContext(this.applicationContext); } } } 2).InitDestroyAnnotationBeanPostProcessor 用于处理 @PostConstruct 、@PreDestroy 注解。 查看 InitDestroyAnnotationBeanPostProcessor 源码, postProcessBeforeInitialization 函数代码,如下: @Override public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException { LifecycleMetadata metadata = findLifecycleMetadata(bean.getClass()); try { metadata.invokeInitMethods(bean, beanName); } catch (InvocationTargetException ex) { throw new BeanCreationException(beanName, "Invocation of init method failed", ex.getTargetException()); } catch (Throwable ex) { throw new BeanCreationException(beanName, "Failed to invoke init method", ex); } return bean; } 使用 findLifecycleMetadata(bean.getClass()); 获取所有的生命周期函数,通过 metadata.invokeInitMethods(bean, beanName); 进行代理调用对应的生命周期函数 public void invokeInitMethods(Object target, String beanName) throws Throwable { Collection<LifecycleElement> initMethodsToIterate = (this.checkedInitMethods != null ? this.checkedInitMethods : this.initMethods); if (!initMethodsToIterate.isEmpty()) { boolean debug = logger.isDebugEnabled(); for (LifecycleElement element : initMethodsToIterate) { if (debug) { logger.debug("Invoking init method on bean '" + beanName + "': " + element.getMethod()); } element.invoke(target); } }} 3).BeanValidationPostProcessor 数据校验 用于进行数据校验,查看 BeanValidationPostProcessor的源码 postProcessBeforeInitialization 和 postProcessAfterInitialization 代码如下,进行 doValidate()函数进行数据校验。 @Override public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException { if (!this.afterInitialization) { doValidate(bean); } return bean; } @Override public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException { if (this.afterInitialization) { doValidate(bean); } return bean; } -END- 可以关注我的公众号,免费获取价值1980元学习资料 点击“在看”,学多少都不会忘~ 本文分享自微信公众号 - 阿聪的全栈之路(gh_ffab7c84fb0c)。如有侵权,请联系 support@oschina.cn 删除。本文参与“OSC源创计划”,欢迎正在阅读的你也加入,一起分享。

优秀的个人博客,低调大师

PowerToys v0.21.1 发布,微软开发的免费实用工具集

PowerToys v0.21.1现已发布,具体更新内容如下: PT Run: 删除了 action keywords 中对空格的需求。这意味着用户现在可以输入>ipconfig 图标缓存已修复,现在带有彩色图标 通过 ClearType 改进了字体渲染 Result速度提高 支持 URLs 修正错误,包括计算错误 FancyZone: Win+Arrow key 是基于区域矩形的方向 修正错误 Runner: 修复了从非管理员帐户提升的 Toast 通知 Shortcut Guide: 改进了 vkey 捕获功能,可以解决某些用例无法显示的问题 SVG in File Explorer: 嵌入式图像标签现在将在资源管理器中呈现 Color Picker: 修复了通过 false positive keystrokes 启动的错误 Accessibility: Settings、PT Run 和 KBM正在进行改进 Localization: Pipeline 现已设置完毕,即将在所有实用程序上进行完整的 E2E 传递。 Dev quality of life improvements: 继续减少警告计数。此版本已删除〜80条 启用了 StyleCop 的E2E 在 E2E 中开始添加 FxCop 更新说明:https://github.com/microsoft/PowerToys/releases/tag/v0.21.1

优秀的个人博客,低调大师

PowerToys v0.20.1 发布,微软开发的免费实用工具集

PowerToys v0.20.1现已发布,这是一个补丁版本,修复了 0.20 版本中对稳定性很重要的几个项目。 具体更新内容如下: #5400-PT Run没有正确处置,并且处置后的对象被试图访问。 #5555-ColorPicker 尝试始终重新加载错误的设置文件 #5540-FancyZones 尝试拖动一个elevated window #5484-安装程序无法安装 .NET Core 更新说明:https://github.com/microsoft/PowerToys/releases/tag/v0.20.1

优秀的个人博客,低调大师

PowerToys v0.20.0 发布,微软开发的免费实用工具集

PowerToys 0.20.0 发布了,此版本添加了新的实用程序,还修复了许多性能和内存问题。 主要更新内容: 添加了拾色器(Color Picker),可使用快捷方式Win+Shift+C 拾取颜色 文件资源管理器 - 现在可以呈现 SVG 图标 FancyZones - 现在可以在拖动窗口的同时按住 Shift+Ctrl 捕捉到 FancyZones 中的任意数量的区域 PT Run - 改进键盘交互 PT Run - 现在检测到新安装的应用程序 PT Run - 许多性能和错误修复 键盘管理器 - 应用级快捷方式 示例:比如 Outlook,将 Ctrl+F 重映射到 F4,现在 Ctrl+F 将显示查找窗口 键盘管理器 - 现在可以将键重新映射到快捷方式以及快捷键 设置 - 改进 OOBE 改进 PowerRename 更新说明:https://github.com/microsoft/PowerToys/releases/tag/v0.20.0

资源下载

更多资源
腾讯云软件源

腾讯云软件源

为解决软件依赖安装时官方源访问速度慢的问题,腾讯云为一些软件搭建了缓存服务。您可以通过使用腾讯云软件源站来提升依赖包的安装速度。为了方便用户自由搭建服务架构,目前腾讯云软件源站支持公网访问和内网访问。

Rocky Linux

Rocky Linux

Rocky Linux(中文名:洛基)是由Gregory Kurtzer于2020年12月发起的企业级Linux发行版,作为CentOS稳定版停止维护后与RHEL(Red Hat Enterprise Linux)完全兼容的开源替代方案,由社区拥有并管理,支持x86_64、aarch64等架构。其通过重新编译RHEL源代码提供长期稳定性,采用模块化包装和SELinux安全架构,默认包含GNOME桌面环境及XFS文件系统,支持十年生命周期更新。

Sublime Text

Sublime Text

Sublime Text具有漂亮的用户界面和强大的功能,例如代码缩略图,Python的插件,代码段等。还可自定义键绑定,菜单和工具栏。Sublime Text 的主要功能包括:拼写检查,书签,完整的 Python API , Goto 功能,即时项目切换,多选择,多窗口等等。Sublime Text 是一个跨平台的编辑器,同时支持Windows、Linux、Mac OS X等操作系统。

WebStorm

WebStorm

WebStorm 是jetbrains公司旗下一款JavaScript 开发工具。目前已经被广大中国JS开发者誉为“Web前端开发神器”、“最强大的HTML5编辑器”、“最智能的JavaScript IDE”等。与IntelliJ IDEA同源,继承了IntelliJ IDEA强大的JS部分的功能。

用户登录
用户注册