当前位置: 首页 > news >正文

Spring MVC源码分析のinit流程

文章目录

  • 前言
  • 一、 init
    • 1.1、createWebApplicationContext
    • 1.2、onRefresh
  • 二、请求处理器
    • 2.1、@RequestMapping
    • 2.2、Controller接口
    • 2.3、HttpRequestHandler接口
    • 2.4、HandlerFunction
  • 三、initHandlerMappings
    • 3.1、getDefaultStrategies
      • 3.1.1、RequestMappingHandlerMapping
      • 3.1.2、BeanNameUrlHandlerMapping
  • 四、initHandlerAdapters
  • 总结


前言

  本篇介绍Spring MVC 的初始化流程,源码的体现是HttpServletBean的init方法。通常Spring MVC的项目,需要打成war包,部署在Tomcat服务器上,也就是Spring MVC通常需要配合Tomcat使用,当然也可以使用Jetty、Undertow 等。
  Spring MVC的源码,主要是分为两部分,第一部分是Tomcat启动时,Spring MVC上下文和容器的初始化。第二部分则是请求到达Tomcat,进行路径映射,转发到DispatcherServlet然后进行处理并且返回的流程。

一、 init

  在Tomcat容器启动时,会进入到HttpServletBeaninit方法,在其中进行初始化的操作:
在这里插入图片描述  initServletBean最终会跳转到FrameworkServletinitWebApplicationContext方法,而在该方法中,最关键的代码是createWebApplicationContext
在这里插入图片描述  在执行完initWebApplicationContext方法刷新容器之后,会将容器对象赋值给webApplicationContext属性,并且执行initFrameworkServlet方法(是一个空实现)。
在这里插入图片描述

1.1、createWebApplicationContext

protected WebApplicationContext createWebApplicationContext(@Nullable ApplicationContext parent) {//首先获取上下文,这里获取到的是XmlWebApplicationContext类型,即解析xml文件的模式Class<?> contextClass = getContextClass();if (!ConfigurableWebApplicationContext.class.isAssignableFrom(contextClass)) {throw new ApplicationContextException("Fatal initialization error in servlet with name '" + getServletName() +"': custom WebApplicationContext class [" + contextClass.getName() +"] is not of type ConfigurableWebApplicationContext");}//尝试通过类型去实例化	XmlWebApplicationContext 容器 //实例化完成后的XmlWebApplicationContext是空的,属性都是默认值ConfigurableWebApplicationContext wac =(ConfigurableWebApplicationContext) BeanUtils.instantiateClass(contextClass);//设置环境wac.setEnvironment(getEnvironment());//设置父容器wac.setParent(parent);//拿到WEB-INF下的spring.xml配置文件String configLocation = getContextConfigLocation();if (configLocation != null) {wac.setConfigLocation(configLocation);}//完成后续容器初始化与刷新操作(配置、监听器、refresh 等)configureAndRefreshWebApplicationContext(wac);return wac;
}

  configureAndRefreshWebApplicationContext方法:

protected void configureAndRefreshWebApplicationContext(ConfigurableWebApplicationContext wac) {// 如果当前容器的 id 还只是默认值(即内存地址),我们给它设置一个更有意义的 idif (ObjectUtils.identityToString(wac).equals(wac.getId())) {// 如果用户自定义设置了 contextId,则使用用户提供的if (this.contextId != null) {wac.setId(this.contextId);} else {// 否则自动生成一个 id:一般是 application:/contextPath/servletNamewac.setId(ConfigurableWebApplicationContext.APPLICATION_CONTEXT_ID_PREFIX +ObjectUtils.getDisplayString(getServletContext().getContextPath()) + '/' + getServletName());}}// 设置容器的 ServletContext,后续会用于获取资源、属性等wac.setServletContext(getServletContext());// 设置 ServletConfig(包含 servlet 的 init 参数等信息)wac.setServletConfig(getServletConfig());// 设置命名空间,一般用于区分不同的 DispatcherServlet 容器wac.setNamespace(getNamespace());// 添加一个监听器,用于监听容器的 refresh 事件(刷新完成时触发)wac.addApplicationListener(new SourceFilteringListener(wac, new ContextRefreshListener()));// 获取当前环境对象,用于管理属性配置(如 application.properties)ConfigurableEnvironment env = wac.getEnvironment();if (env instanceof ConfigurableWebEnvironment) {// 初始化 servletContext 和 servletConfig 中的属性源(property sources)// 这样可以在后续的 Bean 初始化过程中使用 ${...} 占位符引用这些属性((ConfigurableWebEnvironment) env).initPropertySources(getServletContext(), getServletConfig());}// 钩子方法:可以在子类中覆写,对容器进一步自定义处理(例如注册额外 Bean 定义)postProcessWebApplicationContext(wac);// 应用 SpringApplicationContextInitializer(初始化器扩展点)applyInitializers(wac);// 最终刷新容器,触发 Bean 的加载、依赖注入、事件发布等流程wac.refresh();
}

  最终调用的是AbstractApplicationContextrefresh方法,在refresh完成后,XmlWebApplicationContext的属性才有值:
在这里插入图片描述

1.2、onRefresh

  在configureAndRefreshWebApplicationContext方法中,还有一行关键的代码:

wac.addApplicationListener(new SourceFilteringListener(wac, new ContextRefreshListener()));

  这行代码是添加了一个监听器,触发时机是Spring容器refresh完成后。
在这里插入图片描述  最终会跳转到DispatcherServletinitStrategies方法。
在这里插入图片描述在这里插入图片描述  在该方法中,会进行一些初始化操作,其中最重要的是initHandlerMappingsinitHandlerAdapters。在说明这两个方法的逻辑之前,有必要先说明下什么是Handler。

二、请求处理器

  Handler是Spring MVC中的请求处理器,有四种实现:

2.1、@RequestMapping

  RequestMapping是最常见的请求处理器实现,可以加在方法也可以加在类上,如果加在类上,需要配合@Component注解:

@Controller
public class TestController {@Autowiredprivate TestService testService;@RequestMapping(method = RequestMethod.GET, path = "/test")public void test(){System.out.println("test");}
}

2.2、Controller接口

  后三种方式在项目开发中较为少见,第一种是自定义类,实现Controller接口,同时需要将自定义的类标记成bean,然后加上请求的路径(需要加/)

@Component("/test1")
public class MyHandler implements Controller {/*** @param request  current HTTP request * @param response current HTTP response* @return* @throws Exception*/@Overridepublic ModelAndView handleRequest(HttpServletRequest request, HttpServletResponse response) throws Exception {System.out.println("MyHandler implements Controller");return null;}
}

2.3、HttpRequestHandler接口

  自定义一个类,实现HttpRequestHandler接口,和实现Controller接口的区别在于返回值为空。

@Component("/test2")
public class MyHandler1 implements HttpRequestHandler {/*** @param request  current HTTP request * @param response current HTTP response* @throws ServletException* @throws IOException*/@Overridepublic void handleRequest(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {System.out.println("MyHandler1 implements HttpRequestHandler");}
}

2.4、HandlerFunction

  也可以在配置类中注册一个自定义的bean,返回值类型为RouterFunction<ServerResponse>

@ComponentScan("com.zhouyu")
@Configuration
public class AppConfig {@Beanpublic RouterFunction<ServerResponse> person() {return route() .GET("/app/person", request ->ServerResponse.status(HttpStatus.OK).body("Hello GET")).POST("/app/person", request ->ServerResponse.status(HttpStatus.OK).body("Hello POST")).build(); }}

  上述这四种Handler,分别对应Spring MVC源码中DispatcherServlet.properties
在这里插入图片描述

  • @RequestMapping对应BeanNameUrlHandlerMapping
  • Controller接口和HttpRequestHandler接口对应RequestMappingHandlerMapping
  • HandlerFunction对应RouterFunctionMapping

三、initHandlerMappings

  HandlerMapping的作用,就是去记录请求路径上述Handler的对应关系。可以理解底层有一个map,key是请求的路径,value是与之匹配的Handler对象,在容器启动时就将其维护好,避免在请求到达时再去寻找对应的Handler。
  在initHandlerMappings中,首先会去寻找有没有用户自定义的HandlerMapping类型的bean,如果有,就会直接给DispatcherServlethandlerMappings属性赋值为自定义的那个处理器,不会再去用DispatcherServlet.properties中默认的那三个了。
在这里插入图片描述  如果没有自定义的,才会去调用getDefaultStrategiesDispatcherServlet.properties中的:
在这里插入图片描述

3.1、getDefaultStrategies

  在getDefaultStrategies中,首先会拿到DispatcherServlet.properties默认的三个处理器:
在这里插入图片描述  然后会按照顺序循环处理:
在这里插入图片描述  这里要区分不同的情况。

3.1.1、RequestMappingHandlerMapping

  RequestMappingHandlerMapping是针对@RequestMapping注解的情况,处理逻辑的代码在afterPropertiesSet中:
在这里插入图片描述  因为它间接实现了InitializingBean接口,所以处理时机是在bean生命周期中的初始化阶段。我们来看一下RequestMappingHandlerMapping重写的afterPropertiesSet的逻辑,最终调用的是父类AbstractHandlerMethodMappingafterPropertiesSet方法:
在这里插入图片描述  首先会拿到Spring容器中所有bean的名称,然后过滤掉"scopedTarget." 开头的 Bean,最后用当前的候选bean调用processCandidateBean方法:
在这里插入图片描述  在processCandidateBean方法中,会在isHandler中进行判断,当前的bean是否有@Controller注解@RequestMapping注解
在这里插入图片描述在这里插入图片描述如果仅仅在类上加@RequestMapping注解也是不可以的,因为没有@Component,该类就不会被扫描并且注册成bean,也不会进入这一步的逻辑

  对于符合要求的bean,则会进入detectHandlerMethods方法:
在这里插入图片描述  其中的核心方法是getMappingForMethod:

/*** 为指定的方法构建对应的 RequestMappingInfo(映射信息)。* 该方法会综合考虑方法级注解、类级注解,以及类路径前缀等信息。** @param method       要处理的方法(如某个 @RequestMapping 方法)* @param handlerType  方法所属的处理器类* @return 方法对应的 RequestMappingInfo(包含路径、请求方式、参数条件等),如果该方法无映射,则返回 null*/
@Override
@Nullable
protected RequestMappingInfo getMappingForMethod(Method method, Class<?> handlerType) {// 第一步:从方法上提取 @RequestMapping 注解信息(方法级映射)RequestMappingInfo info = createRequestMappingInfo(method);if (info != null) {// 第二步:从类上提取 @RequestMapping 注解信息(类级映射)RequestMappingInfo typeInfo = createRequestMappingInfo(handlerType);// 如果类级映射不为空,则与方法级映射合并,形成完整映射if (typeInfo != null) {info = typeInfo.combine(info);}// 第三步:获取路径前缀(比如配置了统一前缀 "/api"),追加到映射路径中String prefix = getPathPrefix(handlerType);if (prefix != null) {// 构造前缀映射信息,再与原映射 info 合并info = RequestMappingInfo.paths(prefix).options(this.config)  // 使用当前的 config(包含匹配规则等).build().combine(info);}}// 返回最终合并后的 RequestMappingInforeturn info;
}

  在该方法中会解析@RequestMapping 注解信息:
在这里插入图片描述  最终返回到detectHandlerMethods方法的是一个Map<Method, T> methodskey是当前的方法对象,value是注解的信息
在这里插入图片描述  然后会遍历这个map:
在这里插入图片描述  从上图中的registerHandlerMethod进入MappingRegistryregister方法,在该方法中有两个重要的map:

  • pathLookup:key存放了请求路径,value存放了注解的信息。

在这里插入图片描述

  • registry:key存放了注解的对象,value存放了方法的信息。
    在这里插入图片描述
      到这一步为止就完成了路径与方法的映射关系

3.1.2、BeanNameUrlHandlerMapping

  BeanNameUrlHandlerMapping处理的逻辑在setApplicationContext方法中。
  因为BeanNameUrlHandlerMapping间接实现了ApplicationContextAware接口,所以处理时机是在bean生命周期中的初始化前。
  会调用initApplicationContext方法:
在这里插入图片描述  其核心方法是AbstractDetectingUrlHandlerMappingdetectHandlers,同样会先获取容器中所有的bean:
在这里插入图片描述  然后遍历这些bean的名称,找到以"/“开头的(之前说明过,自定义bean实现HttpRequestHandler接口或Controller接口,需要指定bean的名称,并且在最前面加上”/"代表路径)
在这里插入图片描述在这里插入图片描述在这一步注册映射关系

  调用到registerHandler方法,首先根据bean的名称获取到bean:
在这里插入图片描述

  • handlerMap:key存放访问路径,value存放类对象。

在这里插入图片描述  最终handlerMap存放了映射关系:
在这里插入图片描述

四、initHandlerAdapters

  initHandlerAdapters是用于初始化DispatcherServlethandlerAdapters属性的方法。initHandlerAdapters() 的作用就是准备好一组 能够适配各种Controller类型的执行器,让DispatcherServlet能够在处理请求时找到并调用合适的控制器逻辑。
  和initHandlerMappings一样,首先会去找是否有自定义的HandlerAdapter,如果有,就用自定义的,并且不会用默认的了

在这里插入图片描述getDefaultStrategies 同样是去DispatcherServlet.properties中找默认的策略

在这里插入图片描述四种策略,分别对应<二>中的四种方式

  和initHandlerMappings一样,最终同样会去遍历DispatcherServlet.properties中的策略,然后走创建各自bean的流程。
在这里插入图片描述  最终得到四个实例,因为在案例中四种请求方式都有。
在这里插入图片描述  这里也是适配器模式的体现,四种HandlerAdapter都实现了HandlerAdapter适配器接口,真正如何适配的,会在doDispatch(处理请求)的源码中有所体现。
在这里插入图片描述在这里插入图片描述

总结

  Spring MVC在解析web.xml,调用DispatcherServletinit方法时,完成的主要工作可以分为两部分:

  • 添加一个监听器,用于监听容器的 refresh 事件(刷新完成时触发)。
  • 创建Spring容器,调用refresh方法。

  其中Spring容器refresh 完成后,最终会被监听到,并且调用DispatcherServletinitStrategies方法,在该方法中会完成HandlerMappings和HandlerAdapters的初始化。HandlerMappings用于保存路径和方法的映射关系,而HandlerAdapters让DispatcherServlet能够在处理请求时找到并调用合适的控制器逻辑。
  HandlerMappings和HandlerAdapters的初始化,本质都是实例化DispatcherServlet.properties配置文件中定义的bean名称。在各自的生命周期中进行方法回调。BeanNameUrlHandlerMapping处理的逻辑在setApplicationContext方法中,RequestMappingHandlerMapping处理逻辑的代码在afterPropertiesSet中。

相关文章:

Spring MVC源码分析のinit流程

文章目录 前言一、 init1.1、createWebApplicationContext1.2、onRefresh 二、请求处理器2.1、RequestMapping2.2、Controller接口2.3、HttpRequestHandler接口2.4、HandlerFunction 三、initHandlerMappings3.1、getDefaultStrategies3.1.1、RequestMappingHandlerMapping3.1.…...

【后端开发】go-zero微服务框架实践(goland框架对比,go-zero开发实践,文件上传问题优化等等)

【后端开发】go-zero微服务框架实践&#xff08;goland框架对比&#xff0c;go-zero开发实践&#xff0c;文件上传问题优化等&#xff09; 文章目录 1、go框架对比介绍2、go-zero 微服务开发实践3、go-zero 文件上传问题优化 1、go框架对比介绍 国内开源goland框架对比 1 go-…...

C#程序加密与解密Demo程序示例

目录 一、加密程序功能介绍 1、加密用途 2、功能 3、程序说明 4、加密过程 5、授权的注册文件保存方式 二、加密程序使用步骤 1、步骤一 ​编辑2、步骤二 3、步骤三 4、步骤四 三、核心代码说明 1、获取电脑CPU 信息 2、获取硬盘卷标号 3、机器码生成 3、 生成…...

小程序事件系统 —— 33 事件传参 - data-*自定义数据

事件传参&#xff1a;在触发事件时&#xff0c;将一些数据作为参数传递给事件处理函数的过程&#xff0c;就是事件传参&#xff1b; 在微信小程序中&#xff0c;我们经常会在组件上添加一些自定义数据&#xff0c;然后在事件处理函数中获取这些自定义数据&#xff0c;从而完成…...

深入解析 JavaScript 原型与原型链:从原理到应用

原型和原型链是 JavaScript 中实现对象继承和属性查找的核心机制。为了更深入地理解它们&#xff0c;我们需要从底层原理、实现机制以及实际应用等多个角度进行分析。 1. 原型&#xff08;Prototype&#xff09; 1.1 什么是原型&#xff1f; 每个 JavaScript 对象&#xff08…...

关于AI数据分析可行性的初步评估

一、结论&#xff1a;可在部分环节嵌入&#xff0c;无法直接处理大量数据 1.非本地部署的AI应用处理非机密文件没问题&#xff0c;内部文件要注意数据安全风险。 2.AI&#xff08;指高规格大模型&#xff09;十分适合探索性研究分析&#xff0c;对复杂报告无法全流程执行&…...

回归预测 | Matlab实现GWO-BP-Adaboost基于灰狼算法优化BP神经网络结合Adaboost思想的回归预测

回归预测 | Matlab实现GWO-BP-Adaboost基于灰狼算法优化BP神经网络结合Adaboost思想的回归预测 目录 回归预测 | Matlab实现GWO-BP-Adaboost基于灰狼算法优化BP神经网络结合Adaboost思想的回归预测回归效果基本介绍GWO-BP-Adaboost:基于灰狼算法优化BP神经网络结合Adaboost思想…...

ARM Cortex-M 内存映射详解:如何基于寄存器直接读写 寄存器映射方式编码程序 直接操作硬件寄存器来控制 MCU

ARM Cortex-M 的系统映射空间 ​ 在 STM32 等 ARM Cortex-M 系列 MCU 中&#xff0c;内存地址空间按照 存储功能 进行了严格划分&#xff0c;包括 Flash&#xff08;程序存储&#xff09;、RAM&#xff08;数据存储&#xff09;、外设寄存器&#xff08;GPIO、UART、SPI 等&am…...

深度学习实战车辆目标跟踪与计数

本文采用YOLOv8作为核心算法框架&#xff0c;结合PyQt5构建用户界面&#xff0c;使用Python3进行开发。YOLOv8以其高效的实时检测能力&#xff0c;在多个目标检测任务中展现出卓越性能。本研究针对车辆目标数据集进行训练和优化&#xff0c;该数据集包含丰富的车辆目标图像样本…...

django中视图作用和视图功能 以及用法

在 Django REST Framework(DRF)中,视图(View)是处理 HTTP 请求并返回响应的核心组件。DRF 提供了多种视图类,适用于不同的场景和需求。以下是 DRF 中常见的视图类及其作用、使用方法的详细说明: 一、DRF 视图的分类 DRF 的视图可以分为以下几类: 基于函数的视图(Func…...

【每日学点HarmonyOS Next知识】输入框自动获取焦点、JS桥实现方式、Popup设置全屏蒙版、鼠标事件适配、Web跨域

1、HarmonyOS TextInput或TextArea如何自动获取焦点&#xff1f; 可以使用 focusControl.requestFocus 对需要获取焦点的组件设置焦点&#xff0c;具体可以参考文档&#xff1a; https://developer.huawei.com/consumer/cn/doc/harmonyos-references-V5/ts-universal-attribut…...

【学习思维模型】

学习思维模型 一、理解类模型二、记忆类模型三、解决问题类模型四、结构化学习模型五、效率与习惯类模型六、高阶思维模型七、实践建议八、新增学习思维模型**1. 波利亚问题解决四步法****2. 主动回忆(Active Recall)****3. 鱼骨图(因果图/Ishikawa Diagram)****4. MECE原则…...

MyBatis-Plus分页控件使用及使用过程发现的一个坑

最近维护一个旧项目的时候&#xff0c;出现了一个BUG&#xff0c;经排查后发现是Mybatis-plus分页控件使用的时候需要注意的一个问题&#xff0c;故在本地使用MybatisPlus模拟出现了一下这个问题。 首先&#xff0c;先说一下MyBatis-Plus的使用&#xff1a; 1&#xff09;引入…...

STM32的APB1和APB2的区别

STM32微控制器中的APB1和APB2的区别 STM32微控制器中的APB1和APB2是两种不同的外设总线&#xff0c;主要区别在于时钟速度、连接的外设以及用途。以下是它们的详细对比&#xff1a; 1. 时钟速度 APB1 (Advanced Peripheral Bus 1): 低速总线&#xff0c;时钟频率通常为系统时钟…...

JS一些小知识点

一、|| 运算符 plain this.ctx.body { type: type || 0, // ||在此处用法用于默认值填充&#xff0c;判断是否传参或该值是否存在&#xff0c;如果不存在就使用||后买你的值作为默认值 code: code || 0, msg: msg || SUCCESS, data: data || {}, ...others }; 二、trim() 方…...

手写Tomcat:实现基本功能

首先&#xff0c;Tomcat是一个软件&#xff0c;所有的项目都能在Tomcat上加载运行&#xff0c;Tomcat最核心的就是Servlet集合&#xff0c;本身就是HashMap。Tomcat需要支持Servlet&#xff0c;所以有servlet底层的资源&#xff1a;HttpServlet抽象类、HttpRequest和HttpRespon…...

C#变量与变量作用域详解

一、变量基础 1. ‌声明与初始化‌ 声明语法‌&#xff1a;<数据类型> <变量名>&#xff08;如 int age; string name&#xff09;‌初始化要求‌&#xff1a; 1、 类或结构体中的字段变量&#xff08;全局变量&#xff09;‌无需显式初始化‌&#xff0c;默认值…...

SV学习笔记——数组、队列

一、定宽数组 定宽数组是静态变量&#xff0c;编译时便已经确定其大小&#xff0c;其可以分为压缩定宽数组和非压缩定宽数组:压缩数组是定义在类型后面&#xff0c;名字前面;非压缩数组定义在名字后面。Bit [7:0][3:0] name; bit[7:0] name [3:0]; 1.1定宽数组声明 数组的声…...

API调试工具的无解困境:白名单、动态IP与平台设计问题

引言 你是否曾经在开发中遇到过这样的尴尬情形&#xff1a;你打开了平台的API调试工具&#xff0c;准备一番操作&#xff0c;结果却发现根本无法连接到平台&#xff1f;别急&#xff0c;问题出在调试工具本身。今天我们要吐槽的就是那些神奇的开放平台API调试工具&#xff0c;…...

Git清理本地残留的、但已经在服务器上被删除的分支

要筛选出已经被服务器删除的本地分支&#xff0c;并在本地删除这些分支&#xff0c;可以按照以下步骤进行操作&#xff1a; 步骤 1: 获取远程分支信息&#xff0c;确保本地的远程分支信息是最新的&#xff1a; git fetch -p步骤 2: 列出本地分支和远程分支&#xff1a; git …...

java 实现excel文件转pdf | 无水印 | 无限制

文章目录 目录 文章目录 前言 1.项目远程仓库配置 2.pom文件引入相关依赖 3.代码破解 二、Excel转PDF 1.代码实现 2.Aspose.License.xml 授权文件 总结 前言 java处理excel转pdf一直没找到什么好用的免费jar包工具,自己手写的难度,恐怕高级程序员花费一年的事件,也…...

2025 后端自学UNIAPP【项目实战:旅游项目】6、我的收藏页面

代码框架视图 1、先添加一个获取收藏景点的列表请求 【在文件my_api.js文件中添加】 // 引入公共的请求封装 import http from ./my_http.js// 登录接口&#xff08;适配服务端返回 Token&#xff09; export const login async (code, avatar) > {const res await http…...

C++ 求圆面积的程序(Program to find area of a circle)

给定半径r&#xff0c;求圆的面积。圆的面积应精确到小数点后5位。 例子&#xff1a; 输入&#xff1a;r 5 输出&#xff1a;78.53982 解释&#xff1a;由于面积 PI * r * r 3.14159265358979323846 * 5 * 5 78.53982&#xff0c;因为我们只保留小数点后 5 位数字。 输…...

2025季度云服务器排行榜

在全球云服务器市场&#xff0c;各厂商的排名和地位并非一成不变&#xff0c;而是由其独特的优势、战略布局和市场适应性共同决定的。以下是根据2025年市场趋势&#xff0c;对主要云服务器厂商在排行榜中占据重要位置的原因和优势进行深度分析&#xff1a; 一、全球“三巨头”…...

JS设计模式(4):观察者模式

JS设计模式(4):观察者模式 一、引入 在开发中&#xff0c;我们经常会遇到这样的场景&#xff1a;一个对象的状态变化需要自动通知其他对象&#xff0c;比如&#xff1a; 电商平台中&#xff0c;商品库存变化时需要通知所有订阅该商品的用户&#xff1b;新闻网站中&#xff0…...

iOS性能调优实战:借助克魔(KeyMob)与常用工具深度洞察App瓶颈

在日常iOS开发过程中&#xff0c;性能问题往往是最令人头疼的一类Bug。尤其是在App上线前的压测阶段或是处理用户反馈的高发期&#xff0c;开发者往往需要面对卡顿、崩溃、能耗异常、日志混乱等一系列问题。这些问题表面上看似偶发&#xff0c;但背后往往隐藏着系统资源调度不当…...

TSN交换机正在重构工业网络,PROFINET和EtherCAT会被取代吗?

在工业自动化持续演进的今天&#xff0c;通信网络的角色正变得愈发关键。 2025年6月6日&#xff0c;为期三天的华南国际工业博览会在深圳国际会展中心&#xff08;宝安&#xff09;圆满落幕。作为国内工业通信领域的技术型企业&#xff0c;光路科技&#xff08;Fiberroad&…...

在树莓派上添加音频输入设备的几种方法

在树莓派上添加音频输入设备可以通过以下步骤完成&#xff0c;具体方法取决于设备类型&#xff08;如USB麦克风、3.5mm接口麦克风或HDMI音频输入&#xff09;。以下是详细指南&#xff1a; 1. 连接音频输入设备 USB麦克风/声卡&#xff1a;直接插入树莓派的USB接口。3.5mm麦克…...

针对药品仓库的效期管理问题,如何利用WMS系统“破局”

案例&#xff1a; 某医药分销企业&#xff0c;主要经营各类药品的批发与零售。由于药品的特殊性&#xff0c;效期管理至关重要&#xff0c;但该企业一直面临效期问题的困扰。在未使用WMS系统之前&#xff0c;其药品入库、存储、出库等环节的效期管理主要依赖人工记录与检查。库…...

react更新页面数据,操作页面,双向数据绑定

// 路由不是组件的直接跳转use client&#xff0c;useEffect&#xff0c;useRouter&#xff0c;需3个结合&#xff0c; use client表示客户端 use client; import { Button,Card, Space,Tag,Table,message,Input } from antd; import { useEffect,useState } from react; impor…...