一起学SF框架系列5.8-spring-Beans-注解bean解析4-bean解析

前面三节主要讲了如何加载注解Bean的BeanDefinition,执行环节是在DefaultBeanDefinitionDocumentReader.parseBeanDefinitions中用BeanDefinitionParserDelegate.parseCustomElement(ele)加载的,实际上没对注解真正进行解析。本节主要讲述注解bean如何被正确解析。

解析对象

用解析器对含配置类注解的java文件进行解析,例如:

@Configuration
public class AppConfig {@Beanpublic MyServiceImpl myService() {return new MyServiceImpl();}...
}
@Component/@Controller/@Service/@Repository
public class ComponentClass{...
}

解析入口

真正解析应用注解bean是在应用容器刷新时,入口过程:
在这里插入图片描述

解析过程

PostProcessorRegistrationDelegate.invokeBeanFactoryPostProcessors( ConfigurableListableBeanFactory beanFactory, List beanFactoryPostProcessors)

BeanFactoryPostProcessor执行过程:遍历执行处理器的postProcessBeanDefinitionRegistry方法。

	public static void invokeBeanFactoryPostProcessors(ConfigurableListableBeanFactory beanFactory, List<BeanFactoryPostProcessor> beanFactoryPostProcessors) {/* 警告:尽管看起来可以很容易地重构此方法的主体,以避免使用多个循环和多个列表,但使用多个列表和多次传递处理器名称是有意的。我们必须确保遵守PriorityOrdered和Ordered处理顺序。具体来说,我们决不能使处理器被实例化或以错误的顺序在applicationContext中注册。*//* 必须优先调用BeanDefinitionRegistryPostProcessors *///存放bean后置处理器名称的集合Set<String> processedBeans = new HashSet<>();//判断传入的beanFactory是否属于BeanDefinitionRegistry类型if (beanFactory instanceof BeanDefinitionRegistry registry) {//创建BeanFactory后置处理器的集合List<BeanFactoryPostProcessor> regularPostProcessors = new ArrayList<>();//创建用于注册BeanDefinition后置处理器的集合List<BeanDefinitionRegistryPostProcessor> registryProcessors = new ArrayList<>();// 遍历BeanFactory后置处理器for (BeanFactoryPostProcessor postProcessor : beanFactoryPostProcessors) {if (postProcessor instanceof BeanDefinitionRegistryPostProcessor registryProcessor) {//如果是BeanDefinition的后置处理器// 注册BeanDefinition(执行入口)registryProcessor.postProcessBeanDefinitionRegistry(registry);//加入到注册BeanDefinition集合registryProcessors.add(registryProcessor);}else {//如果不是BeanDefinition的后置处理器,直接加入BeanFactory后置处理器集合regularPostProcessors.add(postProcessor);}}/*这里不要初始化FactoryBeans:我们需要保持所有常规bean未初始化,以便让bean工厂的后处理器统一处理!按顺序实现PriorityOrdered、Ordered和其他的BeanDefinitionRegistryPostProcessors的处理。*/// 再创建一个用于每个步骤用的注册BeanDefinition后置处理器的集合 List<BeanDefinitionRegistryPostProcessor> currentRegistryProcessors = new ArrayList<>();/* 第一步:优先把PriorityOrdered类型的BeanDefinitionRegistryPostProcessors加入currentRegistryProcessors,并执行解析入口,然后整个加入到registryProcessors  */String[] postProcessorNames =beanFactory.getBeanNamesForType(BeanDefinitionRegistryPostProcessor.class, true, false);for (String ppName : postProcessorNames) {if (beanFactory.isTypeMatch(ppName, PriorityOrdered.class)) {currentRegistryProcessors.add(beanFactory.getBean(ppName, BeanDefinitionRegistryPostProcessor.class));processedBeans.add(ppName);}}// 排序sortPostProcessors(currentRegistryProcessors, beanFactory);// currentRegistryProcessors中的处理器全部加u到registryProcessorsregistryProcessors.addAll(currentRegistryProcessors);//执行registryProcessors里面所有后处理器的方法-postProcessBeanDefinitionRegistryinvokeBeanDefinitionRegistryPostProcessors(currentRegistryProcessors, registry, beanFactory.getApplicationStartup());// 清空currentRegistryProcessorscurrentRegistryProcessors.clear();/* 第二步:把Ordered类型的BeanDefinitionRegistryPostProcessors加入currentRegistryProcessors,,并执行解析入口,然后整个加入到registryProcessors  (处理方式同“第一步”)  */postProcessorNames = beanFactory.getBeanNamesForType(BeanDefinitionRegistryPostProcessor.class, true, false);for (String ppName : postProcessorNames) {if (!processedBeans.contains(ppName) && beanFactory.isTypeMatch(ppName, Ordered.class)) {currentRegistryProcessors.add(beanFactory.getBean(ppName, BeanDefinitionRegistryPostProcessor.class));processedBeans.add(ppName);}}sortPostProcessors(currentRegistryProcessors, beanFactory);registryProcessors.addAll(currentRegistryProcessors);invokeBeanDefinitionRegistryPostProcessors(currentRegistryProcessors, registry, beanFactory.getApplicationStartup());currentRegistryProcessors.clear();/* 第三步:把其它(非PriorityOrdered、Ordered类型)的BeanDefinitionRegistryPostProcessors加入currentRegistryProcessors,,并执行解析入口,然后整个加入到registryProcessors  (处理方式同“第一步”)  */boolean reiterate = true;while (reiterate) {reiterate = false;postProcessorNames = beanFactory.getBeanNamesForType(BeanDefinitionRegistryPostProcessor.class, true, false);for (String ppName : postProcessorNames) {if (!processedBeans.contains(ppName)) {currentRegistryProcessors.add(beanFactory.getBean(ppName, BeanDefinitionRegistryPostProcessor.class));processedBeans.add(ppName);reiterate = true;}}sortPostProcessors(currentRegistryProcessors, beanFactory);registryProcessors.addAll(currentRegistryProcessors);invokeBeanDefinitionRegistryPostProcessors(currentRegistryProcessors, registry, beanFactory.getApplicationStartup());currentRegistryProcessors.clear();}/* 调用BeanDefinitionRegistryPostProcessor所有处理器的回调功能:postProcessBeanFactory */invokeBeanFactoryPostProcessors(registryProcessors, beanFactory);/* 调用BeanFactoryPostProcessor所有处理器的回调功能:postProcessBeanFactory */invokeBeanFactoryPostProcessors(regularPostProcessors, beanFactory);}else {//beanFactory非BeanDefinitionRegistry类型,直接调所有处理器的回调功能:postProcessBeanFactoryinvokeBeanFactoryPostProcessors(beanFactoryPostProcessors, beanFactory);}/* 这里不要初始化FactoryBeans:我们需要保持所有常规bean未初始化,以便让bean工厂后处理器统一处理!*//* 前面处理不能保证beanFactoryPostProcessors已全部处理,再处理一次 */String[] postProcessorNames =beanFactory.getBeanNamesForType(BeanFactoryPostProcessor.class, true, false);/* 把所有的BeanFactoryPostProcessor分成三类:PriorityOrdered、Ordered和其它 */List<BeanFactoryPostProcessor> priorityOrderedPostProcessors = new ArrayList<>();List<String> orderedPostProcessorNames = new ArrayList<>();List<String> nonOrderedPostProcessorNames = new ArrayList<>();for (String ppName : postProcessorNames) {if (processedBeans.contains(ppName)) {// skip - already processed in first phase above// 如果processedBeans已经包含处理器名字,表示已处理,跳过}else if (beanFactory.isTypeMatch(ppName, PriorityOrdered.class)) {priorityOrderedPostProcessors.add(beanFactory.getBean(ppName, BeanFactoryPostProcessor.class));}else if (beanFactory.isTypeMatch(ppName, Ordered.class)) {orderedPostProcessorNames.add(ppName);}else {nonOrderedPostProcessorNames.add(ppName);}}/* 第一步:处理priorityOrderedPostProcessors:排序,并调用回调功能-postProcessBeanFactory */sortPostProcessors(priorityOrderedPostProcessors, beanFactory);invokeBeanFactoryPostProcessors(priorityOrderedPostProcessors, beanFactory);/* 第二步:处理orderedPostProcessors:转处理器名为bean,排序,并调用回调功能-postProcessBeanFactory */List<BeanFactoryPostProcessor> orderedPostProcessors = new ArrayList<>(orderedPostProcessorNames.size());for (String postProcessorName : orderedPostProcessorNames) {orderedPostProcessors.add(beanFactory.getBean(postProcessorName, BeanFactoryPostProcessor.class));}sortPostProcessors(orderedPostProcessors, beanFactory);invokeBeanFactoryPostProcessors(orderedPostProcessors, beanFactory);/* 第三步:处理其它nonOrderedPostProcessors:转处理器名为bean,不需排序,并调用回调功能-postProcessBeanFactory */List<BeanFactoryPostProcessor> nonOrderedPostProcessors = new ArrayList<>(nonOrderedPostProcessorNames.size());for (String postProcessorName : nonOrderedPostProcessorNames) {nonOrderedPostProcessors.add(beanFactory.getBean(postProcessorName, BeanFactoryPostProcessor.class));}invokeBeanFactoryPostProcessors(nonOrderedPostProcessors, beanFactory);/* 清除缓存的合并bean定义,因为后处理程序可能已经修改了原始元数据,例如替换值中的占位符... */beanFactory.clearMetadataCache();}

从技术角度看,主要做如下处理:
1、是BeanDefinitionRegistryPostProcessor,调用回调方法-postProcessBeanDefinitionRegistry
2、所有BeanDefinitionPostProcessor,按顺序调用回调方法-postProcessBeanFactory

解析器

解析器较多,以ConfigurationClassPostProcessor举例说明。

ConfigurationClassPostProcessor

类关系图

在这里插入图片描述

图中可看出ConfigurationClassPostProcessor是BeanDefinitionRegistryPostProcessor,执行入口为:postProcessBeanDefinitionRegistry

postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry)

注册BeanDefinition。

	//过渡方法@Overridepublic void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) {int registryId = System.identityHashCode(registry);if (this.registriesPostProcessed.contains(registryId)) {throw new IllegalStateException("postProcessBeanDefinitionRegistry already called on this post-processor against " + registry);}if (this.factoriesPostProcessed.contains(registryId)) {throw new IllegalStateException("postProcessBeanFactory already called on this post-processor against " + registry);}this.registriesPostProcessed.add(registryId);processConfigBeanDefinitions(registry);}//处理Configuration类的BeanDefinitionpublic void processConfigBeanDefinitions(BeanDefinitionRegistry registry) {List<BeanDefinitionHolder> configCandidates = new ArrayList<>();String[] candidateNames = registry.getBeanDefinitionNames();// 从所有bean中选择有Configuration注解beanfor (String beanName : candidateNames) {BeanDefinition beanDef = registry.getBeanDefinition(beanName);if (beanDef.getAttribute(ConfigurationClassUtils.CONFIGURATION_CLASS_ATTRIBUTE) != null) {// bean已经处理过,直接跳过if (logger.isDebugEnabled()) {logger.debug("Bean definition has already been processed as a configuration class: " + beanDef);}}else if (ConfigurationClassUtils.checkConfigurationClassCandidate(beanDef, this.metadataReaderFactory)) {/* checkConfigurationClassCandidate为true是如下三类:1、@Configuration注解的类;2、@Component、@ComponentScan、@Import、@ImportResource注解的类;3、包含@Bean注解方法的类 另:条件为true会设置BeanDefinition 的属性:CONFIGURATION_CLASS_ATTRIBUTE:1、@Configuration类设置为CONFIGURATION_CLASS_FULL2、非@Configuration类设置为CONFIGURATION_CLASS_LITE*/configCandidates.add(new BeanDefinitionHolder(beanDef, beanName));}}// 没有Configuration注解bean,直接返回if (configCandidates.isEmpty()) {return;}/// 按@Order排序(没有@Order均为最低优先级)configCandidates.sort((bd1, bd2) -> {int i1 = ConfigurationClassUtils.getOrder(bd1.getBeanDefinition());int i2 = ConfigurationClassUtils.getOrder(bd2.getBeanDefinition());return Integer.compare(i1, i2);});// 对应用容器提供bean名称生成器SingletonBeanRegistry sbr = null;if (registry instanceof SingletonBeanRegistry _sbr) {sbr = _sbr;if (!this.localBeanNameGeneratorSet) {BeanNameGenerator generator = (BeanNameGenerator) sbr.getSingleton(AnnotationConfigUtils.CONFIGURATION_BEAN_NAME_GENERATOR);if (generator != null) {this.componentScanBeanNameGenerator = generator;this.importBeanNameGenerator = generator;}}}// 生成环境容器if (this.environment == null) {this.environment = new StandardEnvironment();}/* 解析@Configuration注解的Class */// 生成解析器ConfigurationClassParser ConfigurationClassParser parser = new ConfigurationClassParser(this.metadataReaderFactory, this.problemReporter, this.environment,this.resourceLoader, this.componentScanBeanNameGenerator, registry);// 复制一份候选者用于解析@ConfigurationSet<BeanDefinitionHolder> candidates = new LinkedHashSet<>(configCandidates);// 已解析的ConfigurationClass容器Set<ConfigurationClass> alreadyParsed = new HashSet<>(configCandidates.size());do {StartupStep processConfig = this.applicationStartup.start("spring.context.config-classes.parse");// 解析处理parser.parse(candidates);// 解析之后验证,一般作用为验证@Configuration注解parser.validate();// 获取已经解析完的configClasses Set<ConfigurationClass> configClasses = new LinkedHashSet<>(parser.getConfigurationClasses());// 删除已经解析的configClasses configClasses.removeAll(alreadyParsed);/* 用ConfigurationClassBeanDefinitionReader注册解析出来的配置类及其Bean类的BeanDefinitionReader到BeanFactory */if (this.reader == null) {this.reader = new ConfigurationClassBeanDefinitionReader(registry, this.sourceExtractor, this.resourceLoader, this.environment,this.importBeanNameGenerator, parser.getImportRegistry());}this.reader.loadBeanDefinitions(configClasses);// 加入到已解析configClasses集合中alreadyParsed.addAll(configClasses);processConfig.tag("classCount", () -> String.valueOf(configClasses.size())).end();// 将已经解析过的configClasses从集合中删除,以方便后续添加其它的配置类对象candidates.clear();/* 判断BeanFactory中的BeanDefinition数量是否大于解析前的BeanDefinition数量。如果大于,则重新获取所有BeanDefinition,然后过滤掉已经解析过的alreadyParsedClasses,再次解析加载一次。原因是前面的Reader对象调用loadBeanDefinition方法时可能会在注册中心中再注册额外的bean定义,且是没有解析过的,因此可能会遗漏一些@Configuration配置类,因此这里需要再获取一次,防止新引入的配置类对象发生遗漏;如此反复,直到没有需解析的ConfigurationClass为止*/if (registry.getBeanDefinitionCount() > candidateNames.length) {// 新的全部BeanDefinitionString[] newCandidateNames = registry.getBeanDefinitionNames();// 老的全部BeanDefinitionSet<String> oldCandidateNames = Set.of(candidateNames);// 已解析的ConfigurationClass容器Set<String> alreadyParsedClasses = new HashSet<>();for (ConfigurationClass configurationClass : alreadyParsed) {alreadyParsedClasses.add(configurationClass.getMetadata().getClassName());}/* 找出需解析的ConfigurationClass,只要candidates不为空,就循环解析直至没有新的需解析的ConfigurationClass为止 */for (String candidateName : newCandidateNames) {if (!oldCandidateNames.contains(candidateName)) {// 不是老的BeanDefinitionBeanDefinition bd = registry.getBeanDefinition(candidateName);if (ConfigurationClassUtils.checkConfigurationClassCandidate(bd, this.metadataReaderFactory) &&!alreadyParsedClasses.contains(bd.getBeanClassName())) {// 必须是注解类 且 不包含在已解析的ConfigurationClass容器candidates.add(new BeanDefinitionHolder(bd, candidateName));}}}// 本次全部的BeanDefinition作为下次循环检查的标准candidateNames = newCandidateNames;}}while (!candidates.isEmpty()); // candidates内容是动态变化的// Register the ImportRegistry as a bean in order to support ImportAware @Configuration classes// 注册ImportRegistry bean,支持ImportAware @Configuration类if (sbr != null && !sbr.containsSingleton(IMPORT_REGISTRY_BEAN_NAME)) {sbr.registerSingleton(IMPORT_REGISTRY_BEAN_NAME, parser.getImportRegistry());}// 保存解析器中PropertySourceDescriptors this.propertySourceDescriptors = parser.getPropertySourceDescriptors();if (this.metadataReaderFactory instanceof CachingMetadataReaderFactory cachingMetadataReaderFactory) {// 清除MetadataReaderFactory中的缓存cachingMetadataReaderFactory.clearCache();}}

ConfigurationClassParser

ConfigurationClass 的解析器。

parse(Set configCandidates)

解析注解类。

	public void parse(Set<BeanDefinitionHolder> configCandidates) {for (BeanDefinitionHolder holder : configCandidates) {BeanDefinition bd = holder.getBeanDefinition();try {if (bd instanceof AnnotatedBeanDefinition annotatedBeanDef) {// 注解类的解析parse(annotatedBeanDef.getMetadata(), holder.getBeanName());}else if (bd instanceof AbstractBeanDefinition abstractBeanDef && abstractBeanDef.hasBeanClass()) {// 抽象BeanDefinition解析parse(abstractBeanDef.getBeanClass(), holder.getBeanName());}else {// 其它BeanDefinition解析parse(bd.getBeanClassName(), holder.getBeanName());}}catch (BeanDefinitionStoreException ex) {throw ex;}catch (Throwable ex) {throw new BeanDefinitionStoreException("Failed to parse configuration class [" + bd.getBeanClassName() + "]", ex);}}this.deferredImportSelectorHandler.process();}// 三个解析方法最终都调用processConfigurationClass,只是构建ConfigurationClass方式不同protected final void parse(@Nullable String className, String beanName) throws IOException {Assert.notNull(className, "No bean class name for configuration class bean definition");MetadataReader reader = this.metadataReaderFactory.getMetadataReader(className);processConfigurationClass(new ConfigurationClass(reader, beanName), DEFAULT_EXCLUSION_FILTER);}protected final void parse(Class<?> clazz, String beanName) throws IOException {processConfigurationClass(new ConfigurationClass(clazz, beanName), DEFAULT_EXCLUSION_FILTER);}protected final void parse(AnnotationMetadata metadata, String beanName) throws IOException {processConfigurationClass(new ConfigurationClass(metadata, beanName), DEFAULT_EXCLUSION_FILTER);}

processConfigurationClass(ConfigurationClass configClass, Predicate filter)

	protected void processConfigurationClass(ConfigurationClass configClass, Predicate<String> filter) throws IOException {// 应当跳过的类:基于用@Conditional注解引入的判断条件来判断if (this.conditionEvaluator.shouldSkip(configClass.getMetadata(), ConfigurationPhase.PARSE_CONFIGURATION)) {return;}ConfigurationClass existingClass = this.configurationClasses.get(configClass);if (existingClass != null) {if (configClass.isImported()) {if (existingClass.isImported()) {existingClass.mergeImportedBy(configClass);}// 配置类是import类且已经被导入过了,无需再次解析,直接返回return;}else {// 如果不是import类,且存在新的且没有解析过,则使用新的配置类替换老的this.configurationClasses.remove(configClass);this.knownSuperclasses.values().removeIf(configClass::equals);}}/* 循环调用doProcessConfigurationClass方法解析ConfigurationClass及其继承层次结构 */// 获取ConfigurationClass的Class字节码SourceClass sourceClass = asSourceClass(configClass, filter);do {// 从源码解析。因为配置类可能存在层级,因此该循环是针对配置类的各层级类进行的sourceClass = doProcessConfigurationClass(configClass, sourceClass, filter);}while (sourceClass != null);// 将解析好的配置类放入集合中this.configurationClasses.put(configClass, configClass);}

doProcessConfigurationClass(ConfigurationClass configClass, SourceClass sourceClass, Predicate filter)

通过从源类中读取注释、成员和方法,应用处理并构建完整的ConfigurationClass。

	@Nullableprotected final SourceClass doProcessConfigurationClass(ConfigurationClass configClass, SourceClass sourceClass, Predicate<String> filter)throws IOException {// 解析@Component注解配置类 注1if (configClass.getMetadata().isAnnotated(Component.class.getName())) {// @Component配置类成员中可能嵌套有注解配置类,需递归处理processMemberClasses(configClass, sourceClass, filter);}// 解析@PropertySource 注解配置类-该注解的作用是引入额外的properties文件,解析过后再将文件属性值注入到environment中,如果有相同的则替换新的for (AnnotationAttributes propertySource : AnnotationConfigUtils.attributesForRepeatable(sourceClass.getMetadata(), PropertySources.class,org.springframework.context.annotation.PropertySource.class)) {if (this.propertySourceRegistry != null) {this.propertySourceRegistry.processPropertySource(propertySource);}else {logger.info("Ignoring @PropertySource annotation on [" + sourceClass.getMetadata().getClassName() +"]. Reason: Environment must implement ConfigurableEnvironment");}}/* 解析@ComponentScan注解配置类 */// 获取@ComponentScan注解对应属性(可能多个)Set<AnnotationAttributes> componentScans = AnnotationConfigUtils.attributesForRepeatable(sourceClass.getMetadata(), ComponentScans.class, ComponentScan.class);if (!componentScans.isEmpty() &&!this.conditionEvaluator.shouldSkip(sourceClass.getMetadata(), ConfigurationPhase.REGISTER_BEAN)) {// 遍历处理for (AnnotationAttributes componentScan : componentScans) {// 用componentScanParse扫描解析,每个componentScan都可能扫描到多个BeanDefinitionSet<BeanDefinitionHolder> scannedBeanDefinitions =this.componentScanParser.parse(componentScan, sourceClass.getMetadata().getClassName());// Check the set of scanned definitions for any further config classes and parse recursively if needed/* 对扫描到的BeanDefinition,判断是否ConfigurationClass,是的话直接进行解析 */for (BeanDefinitionHolder holder : scannedBeanDefinitions) {BeanDefinition bdCand = holder.getBeanDefinition().getOriginatingBeanDefinition();if (bdCand == null) {bdCand = holder.getBeanDefinition();}if (ConfigurationClassUtils.checkConfigurationClassCandidate(bdCand, this.metadataReaderFactory)) {// 递归解析扫描到的ConfigurationClassparse(bdCand.getBeanClassName(), holder.getBeanName());}}}}// 解析@Import 注解配置类processImports(configClass, sourceClass, getImports(sourceClass), filter, true);// 解析@ImportResource 注解配置类AnnotationAttributes importResource =AnnotationConfigUtils.attributesFor(sourceClass.getMetadata(), ImportResource.class);if (importResource != null) {String[] resources = importResource.getStringArray("locations");Class<? extends BeanDefinitionReader> readerClass = importResource.getClass("reader");for (String resource : resources) {String resolvedResource = this.environment.resolveRequiredPlaceholders(resource);configClass.addImportedResource(resolvedResource, readerClass);}}// 解析@Bean 方法(对应的是bean)Set<MethodMetadata> beanMethods = retrieveBeanMethodMetadata(sourceClass);for (MethodMetadata methodMetadata : beanMethods) {configClass.addBeanMethod(new BeanMethod(methodMetadata, configClass));}// 在配置类实现的接口上注册默认方法processInterfaces(configClass, sourceClass);//当前类有父类,返回父类if (sourceClass.getMetadata().hasSuperClass()) {String superclass = sourceClass.getMetadata().getSuperClassName();if (superclass != null && !superclass.startsWith("java") &&!this.knownSuperclasses.containsKey(superclass)) {this.knownSuperclasses.put(superclass, configClass);// Superclass found, return its annotation metadata and recursereturn sourceClass.getSuperClass();}}//当前类没有父类,返回nullreturn null;}

注1:@Configuration、@Controller、@Service、@Repository本身就是@Component。@Configuration源代码如下图:
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/7801.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Mysql关于进程中的死锁和解除锁

Mysql 经常会遇到语句或者存储过程长时间没有反应&#xff0c;大概率就是挂掉了&#xff0c;或者死锁了。 可通过如下几种方式来查看当前进程状态 1. 查询数据库所有的进程状态 SHOW PROCESSLIST SELECT * FROM information_schema.PROCESSLIST; 2. 查询在锁的事务 SELECT…

opencv 图像腐蚀膨胀 erode dilate

#include "iostream" #include "opencv2/opencv.hpp" using namespace std; using namespace cv;int main() {Mat img, dst, dstbin, distancetransform,rel, rel2;img imread("m3.jpg");//转为灰度图cvtColor(img, dst, COLOR_BGR2GRAY);//二…

从Vue2到Vue3【五】——新的组件(Fragment、Teleport、Suspense)

系列文章目录 内容链接从Vue2到Vue3【零】Vue3简介从Vue2到Vue3【一】Composition API&#xff08;第一章&#xff09;从Vue2到Vue3【二】Composition API&#xff08;第二章&#xff09;从Vue2到Vue3【三】Composition API&#xff08;第三章&#xff09;从Vue2到Vue3【四】C…

网络通信原理(第十八课)

网络通信原理(第十八课) 4.1 回顾 1.什么是TCP/IP 目前应用广泛的网络通信协议集 国际互联网上电脑相互通信的规则、约定。 2.主机通信的三要素 IP地址:用来标识一个节点的网络地址(区分网络中电脑身份的地址,如人有名字) 子网掩码:配合IP地址确定网络号 IP路由:网…

10分钟内入门 ArcGIS Pro

本文来源&#xff1a;GIS荟 大家好&#xff0c;这篇文章大概会花费你10分钟的时间&#xff0c;带你入门 ArcGIS Pro 的使用&#xff0c;不过前提是你有 ArcMap 使用经验。 我将从工程文件组织方式、软件界面、常用功能、编辑器、制图这5个维度给大家介绍。 演示使用的 ArcGI…

【Nodejs】nodejs内置模块(中)

1.路劲处理模块 path 1.1 模块概览 在nodejs中&#xff0c;path是个使用频率很高&#xff0c;但却让人又爱又恨的模块。部分因为文档说的不够清晰&#xff0c;部分因为接口的平台差异性。将path的接口按照用途归类&#xff0c;仔细琢磨琢磨&#xff0c;也就没那么费解了。 1.…

计算机网络模型

计算机网络模型 网络模型网络模型中各层对应的协议封装与分用TCP/IP协议簇的组成 网络模型 OSI 七层模型 应用层、表示层、会话层、传输层、网络层、数据链路层、物理层 TCP/IP四层模型 应用层、传输层、网络层、网络接口层 TCP/IP五层模型 应用层、传输层、网络层、数据链路…

iOS transform rotate总结

研究了一下transform的旋转设置&#xff0c;调了半天还以为是旋转写错了&#xff0c;发现是两个不同的view对象写错了&#xff0c;不管怎么说&#xff0c;还是记录一下旋转相关的操作吧。 参数都是弧度。 以一个图片来举例。 let img UIImageView.init() img.image UIImage…

k8s集群环境的搭建

1.环境规划 1.1 集群类型 Kubernetes集群大致分为两类&#xff1a;一主多从和多主多从。 一主多从&#xff1a;一个Master节点和多台Node节点&#xff0c;搭建简单&#xff0c;但是有单机故障风险&#xff0c;适合用于测试环境。 多主多从&#xff1a;多台Master和多台Node节点…

ubuntu与windows之间的文件共享

最近在做项目&#xff0c;需要用到ubuntu进行代码编译生成可执行文件&#xff0c;但是我个人阅读和编写代码喜欢用source insight这个软件IDE。安利一下这个软件&#xff0c;阅读代码和编辑代码真的很棒啊&#xff0c;谁用谁知道吧&#xff01; 由于在ubuntu中安装source insig…

RocketMQ基本概念与入门

文章目录 MQ基本结构依赖案例:productConsumer 核心概念1.nameserver2.broker3.主题队列4.queue队列5. 生产者6.消费者分组和生产者分组7.消费点位 MQ基本结构 message: 消息数据对象product: 程序代码,生成消息,发送消息到队列consumer: 程序代码,监听(绑定)队列,获取消息,执行…

全志F1C200S嵌入式驱动开发(spi-nor image制作)

【 声明:版权所有,欢迎转载,请勿用于商业用途。 联系信箱:feixiaoxing @163.com】 一般soc系统里面添加spi-nor flash芯片,特别是对linux soc来说,都是把它当成文件系统来使用的。spi-nor flash和spi-nand flash相比,虽然空间小了点,但是胜在稳定,这是很多工业…

(二)RabbitMQ【安装Erlang、安装RabbitMQ 、账户管理、管控台、Docker安装 】

Lison <dreamlison163.com>, v1.0.0, 2023.06.22 RabbitMQ【安装Erlang、安装RabbitMQ 、账户管理、管控台、Docker安装 】 文章目录 RabbitMQ【安装Erlang、安装RabbitMQ 、账户管理、管控台、Docker安装 】**安装Erlang**安装RabbitMQ账户管理管控台Docker安装RabbitM…

大数据技术之Hive2

目录标题 3、Hive 数据类型3.1 基本数据类型&#xff1a;3.2 集合数据类型&#xff1a;3.3 类型转化 4、DDL数据定义4.1 创建数据库4.2 查询数据库4.3 创建表4.4 管理表4.5 外部表4.6 管理表与外部表的相互转换4.7 分区表4.7.1 分区表基本操作4.7.2 分区表注意事项 4.7 修改表4…

【后端面经】微服务架构(1-4) | 降级:为什么每次大促的时候总是要把退款之类的服务停掉?

文章目录 一、 前置知识1、什么是降级?2、降级的典型应用3、为什么要降级?4、降级的分类5、如何降级?A) 降级的应用场景B) 跨服务降级C) 提供有损服务二、面试环节1、面试准备2、基本流程3、亮点方案A) 读写服务降级写服务B) 快慢路径降级慢路径三、章节汇总 在熔断章…

Flutter Widget Life Cycle 组件生命周期

Flutter Widget Life Cycle 组件生命周期 视频 前言 了解 widget 生命周期&#xff0c;对我们开发组件还是很重要的。 今天会把无状态、有状态组件的几个生命周期函数一起过下。 原文 https://ducafecat.com/blog/flutter-widget-life-cycle 参考 https://api.flutter.dev/f…

chrome解决http自动跳转https问题

1.地址栏输入&#xff1a; chrome://net-internals/#hsts 2.找到底部Delete domain security policies一栏&#xff0c;输入想处理的域名&#xff0c;点击delete。 3.再次访问http域名不再自动跳转https了。

【Linux后端服务器开发】HTTPS协议

目录 一、加密算法 二、中间人攻击 三、CA认证 一、加密算法 HTTPS协议是什么&#xff1f;HTTPS协议也是一个应用层协议&#xff0c;是在HTTP协议的基础上引入了一个加密层。 HTTP协议内容是按照文本的方式明文传输的&#xff0c;这就导致在传输过程中出现一些被篡改的情况…

vray GPU渲染如何设置?最适合 VRay 渲染的 GPU 是什么?

Chaos 提供的 Vray GPU是一个独立的渲染引擎&#xff0c;提供 GPU 硬件加速。它还与 CPU&#xff08;处理器&#xff09;配合使用&#xff0c;并利用 CPU 和 GPU 硬件进行无缝混合渲染。 GPU 渲染使 Vray GPU 渲染引擎能够在系统的 GPU 而不是 CPU 上执行光线跟踪计算。由于 G…

STM32MP157驱动开发——按键驱动(阻塞与非阻塞)

“阻塞与非阻塞 ”机制&#xff1a; 阻塞&#xff1a; 使用 休眠唤醒机制&#xff0c;read函数会休眠&#xff0c;是阻塞的使用 poll 时&#xff0c;如果传入的超时时间不为 0&#xff0c;表示 read函数会休眠&#xff0c;这种访问方法也是阻塞的。 非阻塞 使用 poll 时&am…