【mybatis源码学习】mybatis的参数处理
一、mybatis的参数处理以及参数取值
1、单个参数
mybatis不做任何处理
取值方式:
#{参数名/任意名}
<!-- Employee getEmpById(Integer id); -->
<select id="getEmpById" resultType="com.mxc.bean.Employee">
select * from employee where id=#{id}
</select>
2、多个参数
mybatis会将多个参数自动封装为一个map
key:param1,…,paramN,也可以是0,…,N-1(即参数索引,N是参数的个数)
value:参数值
取值方式:
#{上述的key}
<!-- Employee getEmpByIdAndLastName(Integer id, String lastName); -->
<select id="getEmpByIdAndLastName" resultType="com.mxc.bean.Employee">
select * from employee where id=#{param1} and last_name=#{param2}
</select>
为多个参数指定明确的key
@Param注解可以为参数指定一个明确的key,方便在sql中取参数值取值方式:
#{@Param注解指定的key}
<!-- Employee getEmpByIdAndLastName(@Param("id")Integer id, @Param("lastName")String lastName); -->
<select id="getEmpByIdAndLastName" resultType="com.mxc.bean.Employee">
select * from employee where id=#{id} and last_name=#{lastName}
</select>
参数多时会封装map,为了取参数值方便,使用@Param来指定封装时使用的key
3、参数是一个POJO
若多个参数刚好是一个POJO中的属性值,可以直接传入一个POJO
取值方式:
#{属性名}
<!-- Employee getEmpByIdAndLastName(Employee emp); -->
<select id="getEmpByIdAndLastName" resultType="com.mxc.bean.Employee">
select * from employee where id=#{id} and last_name=#{lastName}
</select>
4、参数是一个Map
若多个参数不是某一个POJO的属性,可以封装为一个Map
取值方式:
#{Map的key}
<!--
Employee getEmpByIdAndLastName(Map<String, Object> map);
map:
Map<String, Object> map = new HashMap<>();
map.put("id", 1);
map.put("lastName", "mxc");
-->
<select id="getEmpByIdAndLastName" resultType="com.mxc.bean.Employee">
select * from employee where id=#{id} and last_name=#{lastName}
</select>
小试牛刀
Employee getEmp(@Param("id")Integer id,String lastName);
// 取值:id=>#{id/param1},lastName=>#{param2}
Employee getEmp(Integer id,@Param("e")Employee emp);
// 取值:id=>#{param1},lastName => #{param2.lastName/e.lastName}
#{}与${}的异同
同:
可以获取map中的值或者pojo对象属性的值
异:
#{}:是以预编译的形式,将参数设置到sql语句中,可以防止sql注入
${}:取出的值直接拼装在sql语句中,会有安全问题
使用场景
大多情况下,取参数的值使用#{}。在原生JDBC不支持占位符的地方可以使用${}。如按年份分表查询:select * from ${year}_salary;
二、mybatis的参数处理的源码实现
1、 内部数据结构
package com.spring.test.service.mybatis; import org.apache.ibatis.builder.ParameterExpression; /**
*
*/
public class paramHandler { public static void main(String[] args) {
paramsTokenHandler();
} /**
* 方法名:List<User> queryList(String name, RowBounds rowBounds, int age, @Param("address") String addresss);
* 参数内部映射:
* MethodSignature.SortedMap<Integer, String> params
* 参数列表的下标->参数名字(如果是@Param,则显示的是注解的value,否则为参数列表当前的除了RowBounds,ResultHandler以外的参数排名次数)
* 0->0
* 2->1
* 3->address
*
* MethodSignature.convertArgsToSqlCommandParam(Object[] args) 返回:ParamMap
*
* ParamMap内部存储如下:
* 参数名字->参数下标
* 0->name的值
* 1->age的值
* address->addresss的值
*
* param1->name的值
* param2->age的值
* param3->addresss的值
*
*/
public static void paramsToParamMap(){ } /**
* 例子:#{checkedTime,jdbcType=BIGINT,typeHandler=com.meituan.payment.fundsgateway.core.model.handler.DateForLongTypeHandler}
*/
public static void paramsTokenHandler(){
String token="checkedTime,jdbcType=BIGINT,typeHandler=com.meituan.payment.fundsgateway.core.model.handler.DateForLongTypeHandler";
ParameterExpression parameterExpression=new ParameterExpression(token);
String name=parameterExpression.get("property");
System.out.println(name);
String jdbcType=parameterExpression.get("jdbcType");
System.out.println(jdbcType);
String typeHandler=parameterExpression.get("typeHandler");
System.out.println(typeHandler);
}
}
2、涉及到的特殊数据结构和算法
org.apache.ibatis.scripting.defaults.DefaultParameterHandler
=>通过sql解析的ParameterMapping循环遍历,从参数列表中获取指定的值,设置至sql语句中的?占位符。
org.apache.ibatis.reflection.MetaObject
=>将方法的参数列表包装成MetaObject,提供统一获取参数值的方法
org.apache.ibatis.reflection.property.PropertyTokenizer
=>根据sql语句占位符#{propertyName}中的propertyName去MetaObject中获取sql语句中的?的值。
org.apache.ibatis.builder.SqlSourceBuilder
org.apache.ibatis.builder.ParameterMappingTokenHandler
org.apache.ibatis.parsing.GenericTokenParser
org.apache.ibatis.builder.ParameterExpression
=>将sql语句中#{}替换成?,并解析出List<ParameterMapping>
【mybatis源码学习】mybatis的参数处理的更多相关文章
- mybatis源码学习(一) 原生mybatis源码学习
最近这一周,主要在学习mybatis相关的源码,所以记录一下吧,算是一点学习心得 个人觉得,mybatis的源码,大致可以分为两部分,一是原生的mybatis,二是和spring整合之后的mybati ...
- mybatis源码学习:一级缓存和二级缓存分析
目录 零.一级缓存和二级缓存的流程 一级缓存总结 二级缓存总结 一.缓存接口Cache及其实现类 二.cache标签解析源码 三.CacheKey缓存项的key 四.二级缓存TransactionCa ...
- mybatis源码学习:基于动态代理实现查询全过程
前文传送门: mybatis源码学习:从SqlSessionFactory到代理对象的生成 mybatis源码学习:一级缓存和二级缓存分析 下面这条语句,将会调用代理对象的方法,并执行查询过程,我们一 ...
- mybatis源码学习:插件定义+执行流程责任链
目录 一.自定义插件流程 二.测试插件 三.源码分析 1.inteceptor在Configuration中的注册 2.基于责任链的设计模式 3.基于动态代理的plugin 4.拦截方法的interc ...
- Mybatis源码学习第六天(核心流程分析)之Executor分析
今Executor这个类,Mybatis虽然表面是SqlSession做的增删改查,其实底层统一调用的是Executor这个接口 在这里贴一下Mybatis查询体系结构图 Executor组件分析 E ...
- Spring mybatis源码篇章-Mybatis的XML文件加载
通过阅读源码对实现机制进行了解有利于陶冶情操,承接前文Spring mybatis源码篇章-Mybatis主文件加载 前话 前文主要讲解了Mybatis的主文件加载方式,本文则分析不使用主文件加载方式 ...
- mybatis源码学习(三)-一级缓存二级缓存
本文主要是个人学习mybatis缓存的学习笔记,主要有以下几个知识点 1.一级缓存配置信息 2.一级缓存源码学习笔记 3.二级缓存配置信息 4.二级缓存源码 5.一级缓存.二级缓存总结 1.一级缓存配 ...
- Mybatis源码学习之parsing包(解析器)(二)
简述 大家都知道mybatis中,无论是配置文件mybatis-config.xml,还是SQL语句,都是写在XML文件中的,那么mybatis是如何解析这些XML文件呢?这就是本文将要学习的就是,m ...
- Mybatis源码学习之整体架构(一)
简述 关于ORM的定义,我们引用了一下百度百科给出的定义,总体来说ORM就是提供给开发人员API,方便操作关系型数据库的,封装了对数据库操作的过程,同时提供对象与数据之间的映射功能,解放了开发人员对访 ...
- Mybatis源码学习第八天(总结)
源码学习到这里就要结束了; 来总结一下吧 Mybatis的总体架构 这次源码学习我们,学习了重点的模块,在这里我想说一句,源码的学习不是要所有的都学,一行一行的去学,这是错误的,我们只需要学习核心,专 ...
随机推荐
- 【转】Go调度器原理浅析
goroutine是golang的一大特色,或者可以说是最大的特色吧(据我了解),这篇文章主要翻译自Morsing的[这篇博客](http://morsmachine.dk/go-scheduler) ...
- SQL进阶系列之9用SQL处理数列
写在前面 关系模型的数据结构里,并没有顺序的概念,但SQL处理有序集合也有坚实的理论基础 生成连续编号 --生成连续编号 CREATE TABLE Digits (digit INTEGER PRIM ...
- Kotlin协程第一个示例剖析及Kotlin线程使用技巧
Kotlin协程第一个示例剖析: 上一次https://www.cnblogs.com/webor2006/p/11712521.html已经对Kotlin中的协程有了理论化的了解了,这次则用代码来直 ...
- git上传者姓名修改
只需要两个指令 git config user.name 和 git config –global user.name 在控制台中输入git config user.name获取当前的操作名称 修改名 ...
- DATAGRID显示序号
SolidBrush b = new SolidBrush(this.dgvDetail.RowHeadersDefaultCellStyle.ForeColor); e.Gra ...
- The Business Of Open Source
http://oss-watch.ac.uk/resources/businessofopensource by Matthew Langham, Indiginox on 3 February 20 ...
- RobotFrameWork框架介绍与安装
一.RobotFrameWork介绍 1.简称RF,基于python研发的一款自动化测试框架.开源.跨平台的测试框架 2.RF是基于RF基金来研发的一款软件,目前已完全能够在python3环境下应用 ...
- Codeforces Round #605 (Div. 3) A. Three Friends(贪心)
链接: https://codeforces.com/contest/1272/problem/A 题意: outputstandard output Three friends are going ...
- SpringBoot学习(五)RSocket和Security
一.RSocket RSocket是一个用于字节流传输的二进制协议.它通过在单个连接上传递异步消息来支持对称交互模型,主要支持的通讯层包括 TCP, WebSockets和Aeron(UDP). RS ...
- LeetCode 1130. Minimum Cost Tree From Leaf Values
原题链接在这里:https://leetcode.com/problems/minimum-cost-tree-from-leaf-values/ 题目: Given an array arr of ...