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

MyBatis 数据表与实体映射的隐藏陷阱

这两天在处理一个线上问题时,发现Mybatis数据表和实体映射的时候会埋一个坑。这个问题看似微小,但却可能在关键时刻给我们带来不小的困扰。接下来,让我们深入剖析这个问题,并探究其发生的根源。

一、问题描述

我们在使用 Mybatis或者Mybatis-plus时,通常会创建一个数据传输对象(DTO)来封装数据库查询结果。假设我们有一个 EmployeeDTO 类,用于存储员工信息:

public class EmployeeDTO{private Integer employeeNo;private Long employeeId;// 其它字段忽略
}

同时,我们有一段 SQL 语句,用于查询员工列表:

<select id="getEmployeeList" resultType="com.demo.api.vo.EmployeeVo">selectemployee_id as employeeId,employee_no as employeeNofrom employee<where><if test="req.employeeId != null">ra.employee_id = #{req.employeeId}</if>and id > 0 and deleted = 0</where>order by id desclimit #{pageNo}, #{pageSize}</select>

最初,数据表中employee_id属性的类型是int, DTO中的employeeId属性类型是Integer。随着业务发展,employeeId 的值超过了 int 类型的最大值,于是我们将 employee_id 字段类型修改为 bigint,DTO 中 employeeId 属性类型也对应修改为 Long。然而,这个看似简单的修改却导致了程序抛出 NumberOutOfRange 异常:

org.springframework.dao.DataIntegrityViolationException: Error attempting to get column 'EmployeeId' from result set.  Cause: java.sql.SQLDataException: Value '2536800000' is outside of valid range for type java.lang.Integer
; Value '2536800000' is outside of valid range for type java.lang.Integer; nested exception is java.sql.SQLDataException: Value '2536800000' is outside of valid range for type java.lang.Integer

接下来,让我们一起剖析Mybatis的关于resultSet处理的源码,探究问题发生的本源。

二、MyBatis 核心方法分析

1. selectList(String statement, Object parameter, RowBounds rowBounds)

  • 作用
    selectList方法主要用于执行数据库查询操作,并返回一个包含查询结果的列表。这个方法在很多场景下都非常有用,例如获取数据库中的多条记录,实现数据的批量查询等。
  • 实现
    在DefaultSqlSession中,selectList方法会调用query(ms, wrapCollection(parameter), rowBounds, Executor.NO_RESULT_HANDLER)方法执行数据库查询操作。
    (中间会有很多拦截器做相关的逻辑处理,如分页的拦截器,sql验证的拦截器等等,后续有时间了出一篇博客来讲解。)

2. query(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler, CacheKey key, BoundSql boundSql)方法

  • 作用
    主要用于执行数据库查询操作,并根据特定的条件决定是从缓存中获取查询结果还是直接从数据库中进行查询。它通过CacheKey是否存在来判断执行哪种查询方式,确保在合适的情况下利用缓存以提高查询效率
  • 实现
    在BaseExecutor类中,query方法的实现逻辑如下:
	list = resultHandler == null ? (List<E>) localCache.getObject(key) : null;if (list != null) {handleLocallyCachedOutputParameters(ms, key, parameter, boundSql);} else {list = queryFromDatabase(ms, parameter, rowBounds, resultHandler, key, boundSql);}

这里先判断结果处理器resultHandler是否为null,如果是,则尝试从本地缓存localCache中根据CacheKey获取查询结果列表。

如果获取到的列表不为null,说明缓存中有可用的结果,此时会调用handleLocallyCachedOutputParameters方法来处理本地缓存中的输出参数。

这里假定list为null。因此,该方法会调用queryFromDatabase(ms, parameter, rowBounds, resultHandler, key, boundSql)方法执行数据库的查询操作。

3. queryFromDatabase(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler, CacheKey key, BoundSql boundSql)方法

  • 作用
    这个方法的主要作用是在缓存中没有可用结果时,发起对数据库的查询操作。它协调了各个组件之间的交互,确保能够从数据库中获取到所需的数据。
  • 实现
    该方法会调用SimpleExecutor#doQuery(ms, parameter, rowBounds, resultHandler, boundSql)方法执行具体的查询逻辑。
    SimpleExecutor是 MyBatis 中的一个执行器,负责实际执行数据库查询操作。

4. doQuery(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler, BoundSql boundSql) 方法

在 MyBatis 框架中,doQuery方法无疑是最核心的方法之一,在数据库查询操作中发挥着至关重要的作用。

  • 作用
    执行数据库查询操作,并返回查询结果的列表。
    它接受多个参数,包括映射语句(MappedStatement)、查询参数、结果集范围、结果处理器和绑定的 SQL 对象等,通过一系列操作创建语句处理器(StatementHandler)、准备数据库语句(Statement)(这个是重中之重,后面所有的方法都会用到这个作为参数),然后执行查询并处理结果。最后,无论查询是否成功,都会确保关闭数据库语句以释放资源。
  • 实现
public <E> List<E> doQuery(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler, BoundSql boundSql) throws SQLException {Statement stmt = null;try {Configuration configuration = ms.getConfiguration();StatementHandler handler = configuration.newStatementHandler(wrapper, ms, parameter, rowBounds, resultHandler, boundSql);stmt = prepareStatement(handler, ms.getStatementLog());return handler.query(stmt, resultHandler);} finally {closeStatement(stmt);}}

该方法会调用PreparedStatementHandler类的query(stmt, resultHandler)方法执行具体的查询操作,具体的处理逻辑在handleResultSets(ps)方法中。

5. handleResultSets(Statement stmt)方法

在 MyBatis 框架中,handleResultSets方法也是最核心的方法之一,在数据库查询结果的处理过程中起着至关重要的作用。

  • 作用
    这个方法主要负责处理数据库查询返回的结果集。它的任务是将数据库中的数据转换为 Java 对象,以便在应用程序中进行进一步的处理和使用
  • 实现
    该方法会调用handleRowValues方法来处理具体每一行的数据。这个设计体现了 MyBatis 在处理复杂数据结构时的精细分工和模块化设计理念。

6. handleRowValues(ResultSetWrapper rsw, ResultMap resultMap, ResultHandler<?> resultHandler, RowBounds rowBounds, ResultMapping parentMapping)方法

  • 作用
    这个方法主要负责处理结果集中的行数据,将其转换为 Java 对象的属性值
    它根据传入的参数,如ResultSetWrapper(封装了结果集的相关信息)、ResultMap(描述了数据库结果集与 Java 对象之间的映射关系)、ResultHandler(用于处理查询结果的处理器)等,来确定如何将数据库中的每一行数据映射到 Java 对象中。
  • 实现
public void handleRowValues(ResultSetWrapper rsw, ResultMap resultMap, ResultHandler<?> resultHandler, RowBounds rowBounds, ResultMapping parentMapping) throws SQLException {if (resultMap.hasNestedResultMaps()) {ensureNoRowBounds();checkResultHandler();handleRowValuesForNestedResultMap(rsw, resultMap, resultHandler, rowBounds, parentMapping);} else {handleRowValuesForSimpleResultMap(rsw, resultMap, resultHandler, rowBounds, parentMapping);}}

该方法首先判断结果映射中是否存在嵌套的结果映射。如果存在嵌套结果映射,会进行一些特殊的处理,如确保没有行范围限制、检查结果处理器等,并调用handleRowValuesForNestedResultMap方法来处理嵌套结果映射的情况。

如果不是嵌套查询,就会调用handleRowValuesForSimpleResultMap方法来处理具体的结果映射。

7. handleRowValuesForSimpleResultMap(rsw, resultMap, resultHandler, rowBounds, parentMapping)方法

  • 作用
    这个方法专注于处理没有嵌套结构的简单结果映射。其主要任务是将数据库查询结果集中的每一行数据,准确地映射到 Java 对象的相应属性上,以实现数据从数据库到应用程序对象的转换。
  • 实现
    调用getRowValue(rsw, discriminatedResultMap, null)方法获取当前行的 Java 对象值。这个方法根据结果映射规则从结果集中读取数据,并进行必要的类型转换和数据验证,将数据转换为 Java 对象的属性值

8. getRowValue(ResultSetWrapper rsw, ResultMap resultMap, String columnPrefix)方法

getRowValue方法也是Mybatis框架中DefaultResultSetHandler类中最核心的方法之一。

  • 作用
    从给定的ResultSetWrapper(包含SQL的相关信息)和ResultMap(包含java DTO的信息)中,根据特定的列前缀,获取结果集中一行数据的值,并进行适当的处理和转换,以便后续能够准确地映射到 Java 对象的属性中。
  • 实现
    调用createResultObject方法,根据ResultSetWrapper、ResultMap和懒加载映射对象以及列前缀,创建一个结果对象。这个步骤是获取行值的核心,它将从结果集中读取数据,并根据结果映射规则进行类型转换和数据验证,创建一个 Java 对象实例。

9. createResultObject(ResultSetWrapper rsw, ResultMap resultMap, List<Class<?>> constructorArgTypes, List constructorArgs, String columnPrefix)方法

  • 作用
    通过综合考虑结果映射、结果对象类型、构造函数映射等多种因素,以确定最佳的方式来实例化结果对象,为后续的数据处理和业务逻辑提供准确的对象表示。
  • 实现
private Object createResultObject(ResultSetWrapper rsw, ResultMap resultMap, List<Class<?>> constructorArgTypes, List<Object> constructorArgs, String columnPrefix)throws SQLException {final Class<?> resultType = resultMap.getType();final MetaClass metaType = MetaClass.forClass(resultType, reflectorFactory);final List<ResultMapping> constructorMappings = resultMap.getConstructorResultMappings();if (hasTypeHandlerForResultObject(rsw, resultType)) {return createPrimitiveResultObject(rsw, resultMap, columnPrefix);} else if (!constructorMappings.isEmpty()) {return createParameterizedResultObject(rsw, resultType, constructorMappings, constructorArgTypes, constructorArgs, columnPrefix);} else if (resultType.isInterface() || metaType.hasDefaultConstructor()) {return objectFactory.create(resultType);} else if (shouldApplyAutomaticMappings(resultMap, false)) {return createByConstructorSignature(rsw, resultType, constructorArgTypes, constructorArgs);}throw new ExecutorException("Do not know how to create an instance of " + resultType);}

会分情况创建结果对象,这里命中的是自动映射,所以会调用createByConstructorSignature方法,这个方法会通过反射根据构造器签名来创建对象,确保数据可以准确地填充到对象的属性中。

10. createByConstructorSignature(ResultSetWrapper rsw, Class<?> resultType, List

  • 作用
    通过反射机制,根据给定的构造器签名来创建 Java 对象。在处理数据库查询结果集到 Java 对象的映射过程中,这个方法确保了能够正确地实例化 Java 对象,使得数据可以准确地填充到对象的属性中。
  • 实现
private Object createByConstructorSignature(ResultSetWrapper rsw, Class<?> resultType, List<Class<?>> constructorArgTypes, List<Object> constructorArgs) throws SQLException {// 使用反射创建Employee对象final Constructor<?>[] constructors = resultType.getDeclaredConstructors();final Constructor<?> defaultConstructor = findDefaultConstructor(constructors);if (defaultConstructor != null) {return createUsingConstructor(rsw, resultType, constructorArgTypes, constructorArgs, defaultConstructor);} else {for (Constructor<?> constructor : constructors) {if (allowedConstructorUsingTypeHandlers(constructor, rsw.getJdbcTypes())) {return createUsingConstructor(rsw, resultType, constructorArgTypes, constructorArgs, constructor);}}}throw new ExecutorException("No constructor found in " + resultType.getName() + " matching " + rsw.getClassNames());}

通过反射获取给定结果类型resultType(Employee类)的所有声明构造器,默认取第一个。然后会调用createUsingConstructor方法来创建对象。

11. createUsingConstructor(ResultSetWrapper rsw, Class<?> resultType, List

createUsingConstructor方法也是 MyBatis 中最核心的方法之一。

  • 作用
    使用特定的构造函数来创建 Java 对象,确保从数据库查询结果集中准确地提取数据并填充到对象的属性中。
  • 实现
private Object createUsingConstructor(ResultSetWrapper rsw, Class<?> resultType, List<Class<?>> constructorArgTypes, List<Object> constructorArgs, Constructor<?> constructor) throws SQLException {boolean foundValues = false;for (int i = 0; i < constructor.getParameterTypes().length; i++) {// parameterType.name = Java.lang.Integer;Class<?> parameterType = constructor.getParameterTypes()[i];// columnName = employeeId// 所以会对不上。String columnName = rsw.getColumnNames().get(i);TypeHandler<?> typeHandler = rsw.getTypeHandler(parameterType, columnName);Object value = typeHandler.getResult(rsw.getResultSet(), columnName);constructorArgTypes.add(parameterType);constructorArgs.add(value);foundValues = value != null || foundValues;}return foundValues ? objectFactory.create(resultType, constructorArgTypes, constructorArgs) : null;}

首先,会遍历给定构造器类的参数列表。对于每个参数,确定其参数类型和对应的数据库列名。在这个过程中,通过反射获取到的Employee类的构造器类,构造器中每个参数的顺序和Employee定义时一致。而rsw就是第4个核心方法中提到的Statement,rsw.getColumnNames()[i]获取的就是SQL中写的字段顺序。(到这里其实已经找到问题所在了)。

在这个过程中,还通过获取构造器类的参数类型和对应的数据库列名,找到合适的类型处理器(employeeId的类型处理器是java.lang.Integer)。类型处理器负责将数据库中的数据转换为 Java 对象参数所需的类型。

然后,将参数类型和转换后的值分别添加到构造器类的参数类型列表和参数值列表中,并更新是否找到值的标志。根据是否找到值的情况,决定是否创建对象并返回。

三、解决方案

找到了问题根源,解决方案也就呼之欲出,解决方案很简单。
调整 SQL 语句中列的顺序: 将 employee_no 列放在 employee_id 列之前,确保 MyBatis 能够按照正确的顺序进行映射,或者调整Employee类中属性定义的顺序。

四、总结

通过深入 Mybatis关于resultSet处理的源码 ,我们发现,Mybatis在处理结果集ResultSet时,会按照构造器类的参数顺序和ResultSet的列顺序构造对象。在上面的情况下,SQL查询结果的列顺序和DTO中属性顺序不匹配,导致类型处理器使用了错误的类型即尝试把一个很大的bigint赋值给Integer类型

解决这个问题的方法很简单,但更重要的是,我们应该从中吸取教训,在日常开发中更加注重细节,避免类似问题的发生。同时,深入理解 Mybatis 的工作原理,也有助于我们更好地掌握和使用 Mybatis 框架。

本文期望能够帮助大家避开 Mybatis数据表与实体映射的 Unexpected 坑的同时,提供一种解决问题的排查思路,这种软实力才是最重要的

相关文章:

MyBatis 数据表与实体映射的隐藏陷阱

这两天在处理一个线上问题时&#xff0c;发现Mybatis数据表和实体映射的时候会埋一个坑。这个问题看似微小&#xff0c;但却可能在关键时刻给我们带来不小的困扰。接下来&#xff0c;让我们深入剖析这个问题&#xff0c;并探究其发生的根源。 一、问题描述 我们在使用 Mybati…...

leetcode-239. 滑动窗口最大值

题目描述 给你一个整数数组 nums&#xff0c;有一个大小为 k 的滑动窗口从数组的最左侧移动到数组的最右侧。你只可以看到在滑动窗口内的 k 个数字。滑动窗口每次只向右移动一位。 返回 滑动窗口中的最大值 。 示例 1&#xff1a; 输入&#xff1a;nums [1,3,-1,-3,5,3,6,…...

springboot项目中开启mybatis的sql日志

在 application.yml 文件中 添加 mybatis-plus 配置&#xff0c;再重启项目&#xff0c;这里用到了mybatis-plus的自带sql日志打印 # application-jdbc.yml mybatis-plus:configuration:log-impl: org.apache.ibatis.logging.stdout.StdOutImpl如果只是用的mybatis的话&#x…...

卷积的计算——nn.Conv2d(Torch.nn里的Convolution Layers模块里的Conv2d类)

**前置知识&#xff1a; 1、张量和通道 张量&#xff1a;多维数组&#xff0c;用来表示数据&#xff08;图像、视频等&#xff09; 通道&#xff1a;图像数据的一部分&#xff0c;表示不同的颜色或特征层 通道只是张量的其中一个维度 以一张RGB图像为例&#xff0c; 该图像…...

确保接口安全:六大方案有效解决幂等性问题

文章目录 六大方案解决接口幂等问题什么是接口幂等&#xff1f;天然幂等不做幂等会怎么样&#xff1f; 解决方案1&#xff09;insert前先select2&#xff09;使用唯一索引3&#xff09;去重表加悲观锁4&#xff09;加乐观锁之版本号机制5&#xff09;使用 Redisson 分布式锁6&a…...

代码随想录算法训练营第二十九天| 93. 复原 IP 地址,78. 子集, 90. 子集 II

93. 复原 IP 地址&#xff0c;78. 子集&#xff0c; 90. 子集 II 93. 复原 IP 地址78. 子集90. 子集 II 93. 复原 IP 地址 有效 IP 地址 正好由四个整数&#xff08;每个整数位于 0 0 0 到 255 255 255之间组成&#xff0c;且不能含有前导 0 0 0&#xff09;&#xff0c;整…...

【WebGis开发 - Cesium】三维可视化项目教程---初始化场景

系列文章目录 【WebGis开发 - Cesium】三维可视化项目教程—图层管理基础【WebGis开发 - Cesium】三维可视化项目教程—视点管理 目录 系列文章目录引言一、Cesium引入项目1.1 下载资源1.2 项目引入Cesium 二、初始化地球2.1 创建基础文件2.1.1 创建Cesium工具方法文件2.1.2 创…...

点云中ICP算法的详解

ICP&#xff08;Iterative Closest Point&#xff09;算法是一种用于刚性点云配准的经典算法。其核心思想是通过迭代地寻找两个点云之间的最近点对&#xff0c;并计算最优的刚性变换&#xff08;包括旋转和平移&#xff09;&#xff0c;使得源点云在目标点云的坐标系下对齐。IC…...

抽象类Abstart Class

抽象类其实就是一种不完全的设计图 必须用abstract修饰 模板方法&#xff1a;建议使用final修饰&#xff0c;不能被重写。...

Redis:通用命令 数据类型

Redis&#xff1a;通用命令 & 数据类型 通用命令SETGETKEYSEXISTSDELEXPIRETTLTYPEFLUSHALL 数据类型 Redis的客户端提供了很多命令用于操控Redis&#xff0c;在Redis中&#xff0c;key的类型都是字符串&#xff0c;而value有多种类型&#xff0c;每种类型都有自己的操作命…...

【Python高级编程】探索Python库:创建引人入胜的交互界面

1.制作交互界面常用到的库 在 Python 中&#xff0c;有多个库可以用于创建交互界面&#xff08;GUI&#xff09;。 以下是一些常用的 Python GUI 库&#xff1a; Tkinter: Python 的标准 GUI 库&#xff0c;通常随 Python 一起安装。简单易用&#xff0c;适合快速开发小型应用…...

OpenCV Canny()函数

OpenCV Canny()函数被用来检测图像物体的边缘。其算法原理如下&#xff1a; 高斯滤波&#xff1a;使用高斯滤波器平滑图像以减少噪声。高斯滤波器是一种线性滤波器&#xff0c;可以消除图像中的高频噪声&#xff0c;同时保留边缘信息。计算梯度强度和方向&#xff1a;使用Sobe…...

Java基础(3)

基本数据类型 Java 中的几种基本数据类型了解么&#xff1f; Java 中有 8 种基本数据类型&#xff0c;分别为&#xff1a; 6 种数字类型&#xff1a; 4 种整数型&#xff1a;byte、short、int、long2 种浮点型&#xff1a;float、double1 种字符类型&#xff1a;char1 种布尔…...

【C语言】VS调试技巧

文章目录 什么是bug什么是调试&#xff08;debug&#xff09;debug和releaseVS调试快捷键监视和内存观察编程常见错误归类 什么是bug bug本意是“昆虫”或“虫子”&#xff0c;现在一般是指在电脑系统或程序中&#xff0c;隐藏着的一些未被发现的缺陷或问题&#xff0c;简称程…...

【华为HCIP实战课程七】OSPF邻居关系排错MTU问题,网络工程师

一、MTU MUT默认1500,最大传输单元,一致性检测 [R3-GigabitEthernet0/0/1]mtu 1503//更改R3的MTU为1503 查看R3和SW1之间的OSPF邻居关系正常: 默认华为设备没有开启MTU一致性检测! [R3-GigabitEthernet0/0/1]ospf mtu-enable //手动开启MTU检测 [SW1-Vlanif30]ospf mtu…...

速盾:休闲类游戏如何选择高防cdn?

休闲类游戏的流行度日益增长&#xff0c;越来越多的玩家在业余时间里选择放松自己&#xff0c;享受游戏带来的乐趣。然而&#xff0c;在休闲类游戏中&#xff0c;网络延迟和游戏载入速度的问题常常会影响到玩家的游戏体验。为了解决这些问题&#xff0c;选择一个高防CDN&#x…...

电脑插上U盘不显示怎么回事?怎么解决?

平时使用电脑的时候经常会使用U盘来传输数据或是备份文件&#xff0c;有时候会遇到一个令头疼的问题&#xff0c;比如&#xff0c;将U盘插入电脑的USB口后&#xff0c;设备却显示不出来。电脑上插入U盘后却不显示会影响我们的正常工作。接下来&#xff0c;我们一起分析一下故障…...

Python 如何使用 SQLAlchemy 进行复杂查询

Python 如何使用 SQLAlchemy 进行复杂查询 一、引言 SQLAlchemy 是 Python 生态系统中非常流行的数据库处理库&#xff0c;它提供了一种高效、简洁的方式与数据库进行交互。SQLAlchemy 是一个功能强大的数据库工具&#xff0c;支持结构化查询语言&#xff08;SQL&#xff09;…...

nginx主配置文件

Nginx的主配置文件nginx.conf&#xff0c;一般定义了Nginx的基本设置和全局配置。下面是对这个配置文件的详细解释&#xff1a; 文件结构 #user nobody; worker_processes 1;#error_log logs/error.log; #error_log logs/error.log notice; #error_log logs/error.log …...

使用数据库:

数据库&#xff1a; 1.为何需要数据库&#xff1f; 存储数据方法 第一种&#xff1a;用大脑记住数据&#xff0c; 第二种&#xff1a;写纸上&#xff0c; 第三种&#xff1a;写在计算机的内存中&#xff0c; 第四种&#xff1a;写出磁盘文件 2.数据库能做什么&#xff1…...

python list, tuple dict,set的区别 以及**kwargs 的基本用法

在python中, list, tuple, dict, set有什么区别, 主要应用在什么样的场景? 定义: list:链表,有序的项目, 通过索引进行查找,使用方括号”[]”; tuple:元组,元组将多样的对象集合到一起,不能修改,通过索引进行查找, 使用括号”()”; dict:字典,字典是一组键(key)和值(value…...

实用生活英语口语学习成人零基础入门柯桥专业外语培训

“秋裤”的英语表达 首先&#xff0c;秋裤肯定不是autumn pants&#xff0c;chill cool就更离谱了&#xff01; 最地道的美语说法一定会用到“thermal”这个单词&#xff1a; ▼ “thermal”的意思是“热的、保温的”&#xff0c;由此延伸出“秋裤、保暖内衣”的表达&#xff…...

FLINK SQL数据类型

Flink SQL支持非常完善的数据类型&#xff0c;以满足不同的数据处理需求。以下是对Flink SQL数据类型的详细归纳&#xff1a; 一、原子数据类型 字符串类型 CHAR、CHAR(n)&#xff1a;定长字符串&#xff0c;n代表字符的定长&#xff0c;取值范围为[1, 2147483647]。如果不指…...

汇编语言教程:打造你的第一款汇编语言小游戏 汇编语言教程攻略

目录 游戏详细简介 完整代码示例&#xff08;不少于70行&#xff09; 如何自学汇编语言游戏开发攻略及功能 游戏详细简介 游戏名称&#xff1a;“太空探险” 游戏简介&#xff1a;这是一款基于x86汇编语言开发的简单2D游戏。在游戏中&#xff0c;玩家扮演一名宇航员&#…...

白色简洁大方公司企业网站源码 WordPress主题2款

WordPress白色简洁大方公司企业网站主题2款 白色整洁风格wordpress主题是一款比较新颖的国际设计范风格 简洁而大方的 WordPress 主题&#xff0c;适合个人博客、企业和工作室用。 完美支持下拉菜单的wordpress企业主题。 wordpress简白企业模板是一款适合企业站以及工作室…...

MinIO分片上传超大文件(纯服务端)

目录 一、MinIO快速搭建1.1、拉取docker镜像1.2、启动docker容器 二、分片上传大文件到MinIO2.1、添加依赖2.2、实现MinioClient2.3、实现分片上传2.3.0、初始化MinioClient2.3.1、准备分片上传2.3.2、分片并上传2.3.2.1、设置分片大小2.3.2.2、分片 2.3.3、分片合并 三、测试3…...

leetcode链表(一)-移除链表元素

题目 t. - 力扣&#xff08;LeetCode&#xff09; 给你一个链表的头节点 head 和一个整数 val &#xff0c;请你删除链表中所有满足 Node.val val 的节点&#xff0c;并返回 新的头节点 。 例1 输入&#xff1a;head [1,2,6,3,4,5,6], val 6 输出&#xff1a;[1,2,3,4,5]…...

python的特殊方法——魔术方法

前言 __init__(self[]) ​编辑 __call__(self [, ...]) __getitem__(self, key) __len__(self) __repr__(self) / __str__(self) __add__(self, other) __radd__(self, other) 参考文献 前言 官方定义好的&#xff0c;以两个下划线开头且以两个下划线结尾来命名的方法…...

深入浅出理解TCP三次握手与四次挥手

目录 引言1.为什么需要三次握手&#xff1f;2. 三次握手的过程3. 为什么需要四次挥手&#xff1f;4. 四次挥手的过程5. 为什么挥手需要四次&#xff0c;而握手只需三次&#xff1f;6. 三次握手与四次挥手的时序图7. TIME_WAIT状态的意义8. 总结9.面试时候问到什么是三次握手和四…...

如何在Windows和Linux查看正在监听的端口和绑定的进程

端口&#xff08;Port&#xff09;和进程&#xff08;Process&#xff09;是计算机网络和操作系统中的重要概念&#xff0c;它们之间有着密切的关系。以下是对这两个概念的详细介绍以及它们之间的关系(附Windows和Linux查看端口和进程的命令)&#xff1a; 端口&#xff08;Por…...