Mapper解析

你有没有好奇过,使用Mybatis时,为什么我们只需定义接口及xml文件即可完成数据库操作?今天这篇文章就带大家一起来探索一下这个过程,我将从Mapper的解析、调用两方面进行叙述,发车~~~

一、Mapper解析

Mapper解析时主要完成两个任务,一是将Mapper注册至Mapper映射表mapperRegistry中,二是将注解sql、XML sql解析为mappedStatements。

因为mybatis在配置mapper扫描时有不同的方式,所以在完成Mapper的解析时也有所不同,但本质上还是不变的。

当按包路径、类路径进行扫描时,解析Mapper配置内容的统一入口为MapperAnnotationBuilder,而按resource、url进行扫描时,解析Mapper配置内容的统一入口则为XMLMapperBuilder。

以下为不同情况的Mapper配置内容解析调用链。

1、按包路径、类路径进行扫描

一旦 Mapper 注册完成,就由 MapperAnnotationBuilder.parse() 接管:既能解析接口上的注解SQL,也能在命名空间一致时加载并解析 XML,二者在底层通过 LanguageDriver 统一收敛为 SqlSource 。

2、按resource、url进行扫描

可以看出,上述两种场景下解析sql内容,前者的入口为MapperAnnotationBuilder,在解析过程中会调用XMLMapperBuilder,而后者则直接调用XMLMapperBuilder进行解析。那这么做是为什么呢?

  • 扫描 package 时,入口是接口而不是 XML 文件。系统只“知道一堆接口”,所以必须以接口为主导,用 MapperAnnotationBuilder 做“总协调”:既解析方法注解,也“按约定”去找同名 XML 并调用 XMLMapperBuilder 解析。也就是说,MapperAnnotationBuilder 既负责解析注解,又以约定方式“发现并解析”同名 XML,最终把所有 SQL 收敛到 MappedStatement 。
  • 扫描 resource 时,入口就是一个已知的 XML 文件。既然资源已经明确,直接构造 XMLMapperBuilder 读取并解析 XML,完全不需要经过接口协调器。也就是说,XMLMapperBuilder 直接解析 XML 中的 SQL,必要时把命名空间绑定到接口,保证执行期代理与语句能对上号。

这下,我们就知道了,当只提供“接口所在包”时,框架需要一个“总协调器”从接口视角整合注解与 XML;而当明确给了“XML 文件路径”时,框架就可以直奔主题进入 XML 解析器。这是 MyBatis 为兼容“注解 Mapper + XML Mapper + 接口配套 XML”三种使用姿势而做出的清晰职责划分。

二、Mapper调用

1、调用时序图

2、组件关系图

三、深度思考

1、接口和 SQL(XML / 注解)如何关联?

MyBatis 在初始化阶段通过以下两种方式完成绑定:

(1) XML 与接口关联:

通过 @MapperScan 注解扫描指定包下的 Mapper 接口,同时读取配置的 XML 文件路径(如 mybatis.mapper-locations=classpath:mapper/*.xml),将 XML 中 namespace(接口全限定名)与接口匹配,id(方法名)与接口方法匹配;

(2) 注解与接口关联:

扫描接口时直接解析方法上的 @Select/@Insert 等注解,无需 XML 文件,SQL 通过注解直接与方法绑定,无需额外的 namespace 配置。

2、Mapper 接口能不能进行方法重载?

不能。

原因是 MyBatis 通过 “方法名” 作为核心标识匹配 SQL(XML 中的 id 或注解绑定的方法),若存在重载方法(方法名相同、参数不同),会导致 SQL 标识冲突,MyBatis 无法区分对应的 SQL,启动时会抛出异常(如 BindingException: Invalid bound statement)。

Element Plus 是一个基于 Vue 3 的组件库,它提供了丰富的组件,包括上传组件Upload)。在使用 Element Plus 的 Upload 组件进行文件上传操作时,如果想要在点击保存按钮后添加一个加载中的指示器(loading),可以通过监听 Upload 组件的 `submit` 事件,并在事件处理函数中修改组件的状态来显示 loading。 下面是一个简单的实现示例: 1. 在组件模板中使用 `el-upload` 组件,并为其添加 `on-submit` 事件监听器。 2. 在数据对象中定义一个 `loading` 状态,用来控制显示或隐藏 loading。 3. 在 `on-submit` 的事件处理函数中,设置 `loading` 状态为 `true` 来显示 loading,然后执行上传逻辑。 4. 上传完成后,记得将 `loading` 状态设置回 `false`,以隐藏 loading。 ```html <template> <el-upload class="upload-demo" action="your-upload-url" :on-submit="handleUpload" > <el-button size="small" type="primary">点击上传</el-button> <div slot="tip" class="el-upload__tip">只能上传jpg/png文件,且不超过500kb</div> </el-upload> </template> <script> export default { data() { return { loading: false, }; }, methods: { handleUpload(file, fileList) { // 显示 loading this.loading = true; // 文件上传逻辑,例如使用 axios 发送请求到服务器 // axios.post('your-upload-url', file).then(response => { // // 上传成功后隐藏 loading // this.loading = false; // }); }, }, }; </script> ``` 在这个例子中,点击上传按钮后,`handleUpload` 方法会被触发,`loading` 状态被设置为 `true`,从而显示一个加载中的指示器。一旦文件上传完成,你需要在上传成功的回调函数中将 `loading` 状态设置为 `false`,以隐藏加载指示器。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值