Spring MVC 源码分析之 AbstractHandlerMethodAdapter#handle 方法

前言:

前面的篇章我们分析了 Spring MVC 工作流程中的 HandlerMapping、HandlerAdapter 的适配过程,以及拦截器的工作流程,本篇我们来分析正真处理业务请求的过程,也就是 ha.handle(processedRequest, response, mappedHandler.getHandler()),其调用的是 AbstractHandlerMethodAdapter#handle 方法。

Spring MVC 知识传送门:

详解 Spring MVC(Spring MVC 简介)

Spring MVC 初始化源码分析

Spring MVC 工作流程源码分析

Spring MVC 源码分析之 DispatcherServlet#getHandler 方法

Spring MVC 源码分析之 DispatcherServlet#getHandlerAdapter 方法

AbstractHandlerMethodAdapter#handle 方法源码分析

AbstractHandlerMethodAdapter#handle 方法本身没有什么逻辑,内部调用了 RequestMappingHandlerAdapter#handleInternal 方法,该方法会对 Request 请求做一些判断,最终调用 RequestMappingHandlerAdapter#invokeHandlerMethod 方法去处理请求。

//org.springframework.web.servlet.mvc.method.AbstractHandlerMethodAdapter#handle
@Nullable
public final ModelAndView handle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {return this.handleInternal(request, response, (HandlerMethod)handler);
}//org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter#handleInternal
protected ModelAndView handleInternal(HttpServletRequest request, HttpServletResponse response, HandlerMethod handlerMethod) throws Exception {//检查请求 主要检查是否是支持的请求 或者 requireSession 不为空 获取不到 session 校验不通过就抛出异常this.checkRequest(request);//ModelAndViewModelAndView mav;//是否是同步会话if (this.synchronizeOnSession) {//获取sessionHttpSession session = request.getSession(false);//session 为空判断if (session != null) {//本质就是获取 session Object mutex = WebUtils.getSessionMutex(session);//加锁synchronized(mutex) {//处理业务方法 也就是 handler 执行请求处理mav = this.invokeHandlerMethod(request, response, handlerMethod);}} else {//没有可用的 session 无需互斥//处理业务方法 也就是 handler 执行请求处理mav = this.invokeHandlerMethod(request, response, handlerMethod);}} else {//无需同步会话//处理业务方法 也就是 handler 执行请求处理mav = this.invokeHandlerMethod(request, response, handlerMethod);}//判断响应是否需要设置缓存  指定时间内可以直接使用缓存if (!response.containsHeader("Cache-Control")) {if (this.getSessionAttributesHandler(handlerMethod).hasSessionAttributes()) {this.applyCacheSeconds(response, this.cacheSecondsForSessionAttributeHandlers);} else {this.prepareResponse(response);}}return mav;
}

RequestMappingHandlerAdapter#invokeHandlerMethod 方法源码分析

RequestMappingHandlerAdapter#invokeHandlerMethod 会创建一个 ServletInvocableHandlerMethod 实例,并给它设置各个组件,并,会创建模型视图容器 ModelAndViewContainer,用于封装视图、数据模型等,同时会对异步请求做一些处理,通过调用ServletInvocableHandlerMethod#invokeAndHandle 方法执行业务,返回 ModelAndView。

//org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter#invokeHandlerMethod
@Nullable
protected ModelAndView invokeHandlerMethod(HttpServletRequest request, HttpServletResponse response, HandlerMethod handlerMethod) throws Exception {//将请求响应封装成 ServletWebRequestServletWebRequest webRequest = new ServletWebRequest(request, response);ModelAndView var15;try {//把所有使用 @InitBinder 注解修饰的方法 包装成 WebDataBinderFactory 对象WebDataBinderFactory binderFactory = this.getDataBinderFactory(handlerMethod);//把所有使用 @ModelAttribute 注解修饰的方法 包装成 ModelFactory 对象ModelFactory modelFactory = this.getModelFactory(handlerMethod, binderFactory);//将 handlerMethod 包装成 ServletInvocableHandlerMethod 对象ServletInvocableHandlerMethod invocableMethod = this.createInvocableHandlerMethod(handlerMethod);//设置参数解析器if (this.argumentResolvers != null) {invocableMethod.setHandlerMethodArgumentResolvers(this.argumentResolvers);}//设置返回值处理组件if (this.returnValueHandlers != null) {invocableMethod.setHandlerMethodReturnValueHandlers(this.returnValueHandlers);}//设置 binderFactoryinvocableMethod.setDataBinderFactory(binderFactory);//设置 parameterNameDiscovererinvocableMethod.setParameterNameDiscoverer(this.parameterNameDiscoverer);//创建模型视图容器对象ModelAndViewContainer mavContainer = new ModelAndViewContainer();//添加重定向相关的参数 flashMap 对象 重定向参数保存在 flashMap 中mavContainer.addAllAttributes(RequestContextUtils.getInputFlashMap(request));//将 @ModelAttribute 注解修饰的方法将对应的属性存放到 mavContainer 的 model 中modelFactory.initModel(webRequest, mavContainer, invocableMethod);//重定向时忽略默认模型 ignoreDefaultModelOnRedirect 默认值 falsemavContainer.setIgnoreDefaultModelOnRedirect(this.ignoreDefaultModelOnRedirect);//处理异步调用 创建一个 AsyncWebRequestAsyncWebRequest asyncWebRequest = WebAsyncUtils.createAsyncWebRequest(request, response);//设置异步调用的 超时时间asyncWebRequest.setTimeout(this.asyncRequestTimeout);//异步请求管理器WebAsyncManager asyncManager = WebAsyncUtils.getAsyncManager(request);//异步任务的执行者asyncManager.setTaskExecutor(this.taskExecutor);//异步请求管理器asyncManager.setAsyncWebRequest(asyncWebRequest);//注册异步请求回调拦截器asyncManager.registerCallableInterceptors(this.callableInterceptors);//注册异步请求结果处理器asyncManager.registerDeferredResultInterceptors(this.deferredResultInterceptors);Object result;//异步请求的处理if (asyncManager.hasConcurrentResult()) {result = asyncManager.getConcurrentResult();mavContainer = (ModelAndViewContainer)asyncManager.getConcurrentResultContext()[0];asyncManager.clearConcurrentResult();LogFormatUtils.traceDebug(this.logger, (traceOn) -> {String formatted = LogFormatUtils.formatValue(result, !traceOn);return "Resume with async result [" + formatted + "]";});invocableMethod = invocableMethod.wrapConcurrentResult(result);}//调用具体的业务方法invocableMethod.invokeAndHandle(webRequest, mavContainer, new Object[0]);//异步请求处理是否开始if (asyncManager.isConcurrentHandlingStarted()) {result = null;return (ModelAndView)result;}//获取 ModelAndView 对象var15 = this.getModelAndView(mavContainer, modelFactory, webRequest);} finally {//请求完成后的一些处理 会将 requestActive 设置为 false 也就是未激活webRequest.requestCompleted();}//返回return var15;
}

RequestMappingHandlerAdapter#getDataBinderFactory 方法源码分析

RequestMappingHandlerAdapter#getDataBinderFactory 方法主要作用是根据 handlerMethod 创建一个 WebDataBinderFactory 对象 。

//org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter#getDataBinderFactory
private WebDataBinderFactory getDataBinderFactory(HandlerMethod handlerMethod) throws Exception {//获取 handlerMethod 的 type Class<?> handlerType = handlerMethod.getBeanType();//从缓存中获取对应的 InitBinder 方法Set<Method> methods = (Set)this.initBinderCache.get(handlerType);//为空判断if (methods == null) {//缓存中没有获取到 去 handlerType 去类中查找所有标注了 @InitBinder 注解的方法 并将其缓存methods = MethodIntrospector.selectMethods(handlerType, INIT_BINDER_METHODS);this.initBinderCache.put(handlerType, methods);}//InvocableHandlerMethod 对象集合List<InvocableHandlerMethod> initBinderMethods = new ArrayList();//从标注了 @ControllerAdvice 注解的类中寻找 @InitBinder 注解修饰的方法 并为其创建 InvocableHandlerMethod 对象this.initBinderAdviceCache.forEach((controllerAdviceBean, methodSet) -> {if (controllerAdviceBean.isApplicableToBeanType(handlerType)) {Object bean = controllerAdviceBean.resolveBean();Iterator var6 = methodSet.iterator();while(var6.hasNext()) {Method method = (Method)var6.next();//为 method 创建 InvocableHandlerMethod 对象  并加入 initBinderMethods 中initBinderMethods.add(this.createInitBinderMethod(bean, method));}}});Iterator var5 = methods.iterator();while(var5.hasNext()) {Method method = (Method)var5.next();Object bean = handlerMethod.getBean();//为 method 创建 InvocableHandlerMethod 对象  并加入 initBinderMethods 中initBinderMethods.add(this.createInitBinderMethod(bean, method));}//使用 InvocableHandlerMethod 集合 创建 WebDataBinderFactory 对象return this.createDataBinderFactory(initBinderMethods);
}

RequestMappingHandlerAdapter#getModelFactory 方法源码分析

RequestMappingHandlerAdapter#getModelFactory方法主要作用是根据 @SessionAttributes 注解和 @ModelAttribute 注解修饰的方法以及 binderFactory 创建一个 ModelFactory 对象 。

//org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter#getModelFactory
private ModelFactory getModelFactory(HandlerMethod handlerMethod, WebDataBinderFactory binderFactory) {//处理 @SessionAttributes 注解SessionAttributesHandler sessionAttrHandler = this.getSessionAttributesHandler(handlerMethod);//获取 handlerMethod 的 type Class<?> handlerType = handlerMethod.getBeanType();//用缓存中获取Set<Method> methods = (Set)this.modelAttributeCache.get(handlerType);if (methods == null) {//缓存为空  去 handlerType 类中寻找所有被 @ModelAttribute  注解修饰的方法methods = MethodIntrospector.selectMethods(handlerType, MODEL_ATTRIBUTE_METHODS);//加入缓存this.modelAttributeCache.put(handlerType, methods);}//InvocableHandlerMethod 集合List<InvocableHandlerMethod> attrMethods = new ArrayList();从标注了 @ControllerAdvice 注解的类中寻找 @ModelAttribute 注解修饰的方法 并为其创建 InvocableHandlerMethod 对象this.modelAttributeAdviceCache.forEach((controllerAdviceBean, methodSet) -> {if (controllerAdviceBean.isApplicableToBeanType(handlerType)) {Object bean = controllerAdviceBean.resolveBean();Iterator var7 = methodSet.iterator();while(var7.hasNext()) {Method method = (Method)var7.next();//为 method 创建 InvocableHandlerMethod 对象  并加入 initBinderMethods 中attrMethods.add(this.createModelAttributeMethod(binderFactory, bean, method));}}});Iterator var7 = methods.iterator();while(var7.hasNext()) {Method method = (Method)var7.next();Object bean = handlerMethod.getBean();//为 method 创建 InvocableHandlerMethod 对象  并加入 initBinderMethods 中attrMethods.add(this.createModelAttributeMethod(binderFactory, bean, method));}//根据 attrMethods binderFactory sessionAttrHandler 创建 ModelFactoryreturn new ModelFactory(attrMethods, binderFactory, sessionAttrHandler);
}

ModelFactory#initModel 方法源码分析

ModelFactory#initModel 方法主要作用是把 @SessionAttributes 注解和 @ModelAttribute 注解修饰方法的参数初始化到容器中。

//org.springframework.web.method.annotation.ModelFactory#initModel
public void initModel(NativeWebRequest request, ModelAndViewContainer container, HandlerMethod handlerMethod) throws Exception {//解析 @SessionAttributes 注解 的参数Map<String, ?> sessionAttributes = this.sessionAttributesHandler.retrieveAttributes(request);//合并 @SessionAttributes 注解修饰方法的参数container.mergeAttributes(sessionAttributes);//调用被 @ModelAttribute 注解修饰的方法this.invokeModelAttributeMethods(request, container);//查找 @SessionAttributes  注解修饰方法的参数Iterator var5 = this.findSessionAttributeArguments(handlerMethod).iterator();while(var5.hasNext()) {String name = (String)var5.next();if (!container.containsAttribute(name)) {//解析参数Object value = this.sessionAttributesHandler.retrieveAttribute(request, name);if (value == null) {throw new HttpSessionRequiredException("Expected session attribute '" + name + "'", name);}//加入到 模型视图容器中container.addAttribute(name, value);}}}

ServletInvocableHandlerMethod#invokeAndHandle 方法源码分析

ServletInvocableHandlerMethod#invokeAndHandle 方法主要是调用了具体的业务方法,然后对返回值进行判断处理,封装返回值到 ModelAndViewContainer 中。

//org.springframework.web.servlet.mvc.method.annotation.ServletInvocableHandlerMethod#invokeAndHandle
public void invokeAndHandle(ServletWebRequest webRequest, ModelAndViewContainer mavContainer, Object... providedArgs) throws Exception {//调用 controller 中的具体方法Object returnValue = this.invokeForRequest(webRequest, mavContainer, providedArgs);//设置响应状态码this.setResponseStatus(webRequest);//返回值判断if (returnValue == null) {//if (this.isRequestNotModified(webRequest) || this.getResponseStatus() != null || mavContainer.isRequestHandled()) {//是否禁用缓存this.disableContentCachingIfNecessary(webRequest);//设置请求已经处理mavContainer.setRequestHandled(true);return;}} else if (StringUtils.hasText(this.getResponseStatusReason())) {//请求有错误原因 设置请求已经处理mavContainer.setRequestHandled(true);return;}//请求有返回值 且没有错误原因 设置请求未处理  继续处理请求 mavContainer.setRequestHandled(false);//断言 returnValueHandlersAssert.state(this.returnValueHandlers != null, "No return value handlers");try {//选择合适的 HandlerMethodReturnValueHandler 处理返回值this.returnValueHandlers.handleReturnValue(returnValue, this.getReturnValueType(returnValue), mavContainer, webRequest);} catch (Exception var6) {if (this.logger.isTraceEnabled()) {this.logger.trace(this.formatErrorForReturnValue(returnValue), var6);}throw var6;}
}

InvocableHandlerMethod#invokeForRequest方法源码分析

InvocableHandlerMethod#invokeForRequest 方法仅仅是解析了方法的参数值,然后调用了 InvocableHandlerMethod#doInvoke 方法,该方法通过桥接方法调用了具体的方法,也就是 Controller 中定义的方法。

//org.springframework.web.method.support.InvocableHandlerMethod#invokeForRequest
@Nullable
public Object invokeForRequest(NativeWebRequest request, @Nullable ModelAndViewContainer mavContainer, Object... providedArgs) throws Exception {//获取方法参数值Object[] args = this.getMethodArgumentValues(request, mavContainer, providedArgs);if (this.logger.isTraceEnabled()) {this.logger.trace("Arguments: " + Arrays.toString(args));}//调用方法return this.doInvoke(args);
}//org.springframework.web.method.support.InvocableHandlerMethod#doInvoke
@Nullable
protected Object doInvoke(Object... args) throws Exception {//底层就是 setAccessible(true)  也就是禁止 java 语言访问检查 如果 false 就是强制 java 语言访问检查ReflectionUtils.makeAccessible(this.getBridgedMethod());try {//获取桥接方法实例 并执行实际方法  其实就是 controller 类中的方法return this.getBridgedMethod().invoke(this.getBean(), args);} catch (IllegalArgumentException var4) {this.assertTargetBean(this.getBridgedMethod(), this.getBean(), args);String text = var4.getMessage() != null ? var4.getMessage() : "Illegal argument";throw new IllegalStateException(this.formatInvokeError(text, args), var4);} catch (InvocationTargetException var5) {Throwable targetException = var5.getTargetException();if (targetException instanceof RuntimeException) {throw (RuntimeException)targetException;} else if (targetException instanceof Error) {throw (Error)targetException;} else if (targetException instanceof Exception) {throw (Exception)targetException;} else {throw new IllegalStateException(this.formatInvokeError("Invocation failure", args), targetException);}}
}

桥接方法:通过判断方法名、参数的个数以及泛型类型参数来获取桥接方法的实际方法(希望后续有时间去研究一下桥接方法)。

RequestMappingHandlerAdapter#getModelAndView 方法源码分析

RequestMappingHandlerAdapter#getModelAndView 主要是从 ModelAndViewContainer 中取出结果,包装成 ModelAndView 并返回给前端控制器 DispatcherServlet,至此整个 Handler 处理器的调用完成,后续进入视图处理阶段。

//org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter#getModelAndView
@Nullable
private ModelAndView getModelAndView(ModelAndViewContainer mavContainer, ModelFactory modelFactory, NativeWebRequest webRequest) throws Exception {//更新 ModelAndViewContainer 中的 model 属性modelFactory.updateModel(webRequest, mavContainer);//请求已经处理 返回 nullif (mavContainer.isRequestHandled()) {return null;} else {//从模型视图容器中取出 modelModelMap model = mavContainer.getModel();//创建 模型视图ModelAndView mav = new ModelAndView(mavContainer.getViewName(), model, mavContainer.getStatus());if (!mavContainer.isViewReference()) {//设置模型视图的 视图mav.setView((View)mavContainer.getView());}//初始 重定向 的参数if (model instanceof RedirectAttributes) {Map<String, ?> flashAttributes = ((RedirectAttributes)model).getFlashAttributes();HttpServletRequest request = (HttpServletRequest)webRequest.getNativeRequest(HttpServletRequest.class);if (request != null) {RequestContextUtils.getOutputFlashMap(request).putAll(flashAttributes);}}//返回return mav;}
}

至此 AbstractHandlerMethodAdapter#handle 方法的调用链路源码分析完毕,这部分也是 Spring MVC 真正调用 Controller 中定义方法的一段源码,希望可以帮助到有需要的小伙伴们。

欢迎提出建议及对错误的地方指出纠正。

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

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

相关文章

【怀旧版】win10中从零开始创建vue2+ElementUI项目

技术栈 node22 vue2 cli2.9 webpack3.6 elementUI2 必读&#xff1a; https://u.rudon.cn/wgz5 > Vue简史https://u.rudon.cn/bvg0 > 使用 nvm 管理不同版本的 node 与 npmhttps://u.rudon.cn/ehn3 > nodejs和vue关系https://u.rudon.cn/qyb4 > vue的vite、v…

Java 应用部署与优化:简单介绍Java应用的部署策略,并讲解一些常用的Java应用性能优化技巧

I. Java 应用部署 A. 容器化部署 Docker 的简介及其优势 Docker是一种开源的容器化技术,它可以将应用及其依赖打包在一起作为一个可运行的独立单元进行运行。Docker的主要优势包括以下几点: 便携性:无论在哪种环境下,只要安装了Docker,就可以运行Docker容器。 一致性:…

ArcGIS教程(05):计算服务区和创建 OD 成本矩阵

准备视图 启动【ArcMap】->双击打开【Exercise05.mxd】->启用【Network Analyst 扩展模块】。前面的文章已经讲过&#xff0c;这里不再赘述。 创建服务区分析图层 1、在 Network Analyst 工具栏上&#xff0c;单击 【Network Analyst】&#xff0c;然后单击【新建服务…

解决安装 WP Super Cache 插件提示 Advanced-Cache.Php 是另一个插件创建的

昨天晚上一个站长求助明月&#xff0c;说是安装 WP Super Cache 插件的时候提示 advanced-cache.php 被占用了&#xff0c;无法完成安装&#xff0c;收到截图看了才明白原来提示的是“advanced-cache.php 文件&#xff0c;由另一个插件或者系统管理员创建的”&#xff0c;如下图…

社交媒体数据恢复:QQ空间

本教程将指导您如何恢复QQ空间中的说说、日志和照片等内容。请注意&#xff0c;本教程不涉及推荐任何数据恢复软件。 一、恢复QQ空间说说 登录您的QQ账号&#xff0c;并进入QQ空间。点击“日志”选项&#xff0c;进入空间日志页面。在空间日志页面&#xff0c;您会看到一个“…

go语言使用model Gorm MySQL查询数据 定时十分钟查询一次 查询十分钟前新建的数据

在Go语言中&#xff0c;使用GORM库与MySQL数据库交互并定时查询数据是常见的需求。以下是一个基本的示例&#xff0c;展示了如何设置定时任务&#xff0c;并使用GORM查询十分钟前新建的数据&#xff1a; 首先&#xff0c;你需要安装GORM和MySQL驱动&#xff1a; bash go get -…

Practicing Version Control

Part A 新建一个文件夹Git&#xff0c;然后进入文件夹: Windows: 在这个文件夹中右键 > Open Git Bash hereMac: 运行终端&#xff0c;打cd 空格&#xff0c;然后将文件夹拖入终端&#xff0c;按 return 从 Github 上复制 HTTPS&#xff0c;然后 git clone https://githu…

16-云原生监控体系-rabbitmq_exporter监控 RabbitMQ-[部署Dashborad告警规则实战]

文章目录 1. 二进制方式部署1.1. 二进制包下载和部署1.2. 配置1.2.1. 可用的环境变量1.2.2. 使用变量2. docker-compose 方式部署3. 配置到 Prometheus3. Metrics3.1. 全局3.2. 基础信息3.3. Queues3.3.1 Queues - Gauge3.3.2. Queues - Counter

Fish语言编程:探索一种全新的编程范式

Fish语言编程&#xff1a;探索一种全新的编程范式 在浩瀚的编程海洋中&#xff0c;Fish语言以其独特的魅力吸引着无数编程爱好者的目光。作为一种全新的编程范式&#xff0c;Fish语言以其简洁、直观和强大的特性&#xff0c;为编程世界带来了新的活力。本文将从四个方面、五个…

数据库(12)——DQL聚合查询

常见的聚合函数 将一列数据作为一个整体&#xff0c;进行纵向计算。 函数功能count统计数量max最大值min最小值avg平均值sum求和 语法 SELECT 聚合函数 &#xff08;字段列表&#xff09;FROM 表名; 示例 这是我们的原始表&#xff1a; 求人物总数 select count(id) from in…

2024-06-01 问AI: 在大语言模型中,什么是single agent?

文心一言 在大语言模型&#xff08;LLM&#xff09;的语境中&#xff0c;特别是在涉及多智能体&#xff08;Multi-Agent&#xff09;系统的讨论中&#xff0c;Single-Agent&#xff08;单智能体&#xff09;指的是整个系统中仅设计了一个智能体来负责处理工作流中的各个环节。…

“开源与闭源:AI大模型发展的未来之路“

文章目录 每日一句正能量前言数据隐私开源大模型与数据隐私闭源大模型与数据隐私数据隐私保护的共同考虑结论 商业应用开源大模型的商业应用优势&#xff1a;开源大模型的商业应用劣势&#xff1a;闭源大模型的商业应用优势&#xff1a;闭源大模型的商业应用劣势&#xff1a;商…

Navicat使用ssh隧道连接mysql数据库

转载请标明出处&#xff1a;http://blog.csdn.net/donkor_/article/details/139352748 文章目录 前言新建连接MySql,填写ssh隧道信息方式1&#xff1a;使用密码方式连接方式二&#xff1a;使用密钥方式连接 填写常规信息总结 前言 使用ssh隧道连接数据库&#xff0c;方便本机…

docker和虚拟机的异同

Docker 和虚拟机&#xff08;Virtual Machine, VM&#xff09;都是用于隔离和运行应用程序的技术&#xff0c;但它们在实现方式、性能和使用场景上有显著的差异。以下是它们的主要异同点&#xff1a; 相同点 隔离性&#xff1a;两者都提供隔离环境&#xff0c;确保应用程序运…

2024抖音流量认知课:掌握流量底层逻辑,明白应该选择什么赛道 (43节课)

课程下载&#xff1a;https://download.csdn.net/download/m0_66047725/89360865 更多资源下载&#xff1a;关注我。 课程目录 01序言&#xff1a;拍前请看.mp4 02抖音建模逻辑1.mp4 03抖音标签逻辑2.mp4 04抖音推流逻辑3.mp4 05抖音起号逻辑4.mp4 06养号的意义.mp4 0…

Linux 多台机器之间的免密登录设置

生成公钥私钥 ssh-keygen -t rsa 一路回车即可 将公钥发送给需要免密的机群 # 格式&#xff1a;ssh-copy-id -i ~/.ssh/id_rsa.pub 用户名机群地址 ssh-copy-id -i ~/.ssh/id_rsa.pub hadoop192.168.0.1

【经典排序算法】堆排序(精简版)

什么是堆排序&#xff1a; 堆排序(Heapsort)是指利用堆&#xff08;完全二叉树&#xff09;这种数据结构所设计的一种排序算法&#xff0c;它是选择排序的一种。需要注意的是排升序要建大堆&#xff0c;排降序建小堆。 堆排序排序的特性总结&#xff1a; 1. 堆排序使用堆来选数…

flink left join消费kafka数据

left join会产生回车流数据 在控制台数据 import com.sjfood.sjfood.gmallrealtime.app.BaseSQLAPP; import com.sjfood.sjfood.gmallrealtime.util.SQLUtil; import org.apache.flink.streaming.api.environment.StreamExecutionEnvironment; import org.apache.flink.table.…

电脑设置密码怎么设置?让你的电脑更安全!

在如今信息化的社会中&#xff0c;保护个人电脑的安全至关重要。设置密码是最基本的电脑安全措施之一&#xff0c;它可以有效防止未经授权的访问和保护个人隐私&#xff0c;可是电脑设置密码怎么设置&#xff1f;本文将介绍三种设置电脑密码的方法&#xff0c;帮助您加强电脑的…