mybatis源码解读之executor包语句处理功能


Posted in Java/Android onFebruary 15, 2022

1.mybatis对多语句类型的支持

mybatis映射文件中传参数,主要用到#{} 或者 ${}.

#{}:表示使用这种符号的变量会以预编译的形式赋值到sql片段中。

${}:表示使用这种符号的变量会以字符串的形式直接插到sql片段中。

mybatis中支持三种语句类型,不同语句类型支持的变量符号不同。mybatis的三种类型如下:

  • STATEMENT:这种语句类型中,只会对sql片段进行简单的字符串拼接。只支持使用${}.
  • PREPARED:这种语句中会先对sql片段进行字符串拼接,然后再对sql片段进行赋值。可以使用#{}和${}.
  • CALLABLE:这种语句用了实现执行过程的调用,会先对sql片段进行字符串拼接,然后对sql片段进行赋值。可以使用#{}和${}.

2.mybatis的语句处理功能

statement子包负责提供语句处理功能,其中StatementHandler是语句功能类的父接口,RoutingStatementHandler类是一个代理类,它能够根据传入的MappedStatement对象的具体类型选中一个具体的被代理对象,然后将所有实际操作都委托给被代理对象。所以RoutingStatementHandler类提供的是路由功能,而路由选择的依据就是语句类型。

public class RoutingStatementHandler implements StatementHandler {

  // 根据语句类型选取出的被代理类的对象
  private final StatementHandler delegate;

  public RoutingStatementHandler(Executor executor, MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler, BoundSql boundSql) {
    // 根据语句类型选择被代理对象
    switch (ms.getStatementType()) {
      case STATEMENT:
        delegate = new SimpleStatementHandler(executor, ms, parameter, rowBounds, resultHandler, boundSql);
        break;
      case PREPARED:
        delegate = new PreparedStatementHandler(executor, ms, parameter, rowBounds, resultHandler, boundSql);
        break;
      case CALLABLE:
        delegate = new CallableStatementHandler(executor, ms, parameter, rowBounds, resultHandler, boundSql);
        break;
      default:
        throw new ExecutorException("Unknown statement type: " + ms.getStatementType());
    }
  }
}

BaseStatementHandler作为三个实现类的父类,提供了实现类的公共方法。并且BaseStatementHandler类使用的模板模式在prepare方法中定义了整个方法的框架,然后将一些与子类相关的操作交给三个子类处理。

SimpleStatementHandler类、PreparedStatementHandler类和CallableStatementHandler类是三个真正的statement处理器,分别处理statement、preparedStatementCallableStatement对象。通过其中的parameterize方法可以看出三个Statement处理器的不同。

SimpleStatementHandlerparameterize方法的实现为空,因为它只需要完成字符串替换即可,不需要进行参数处理

public class SimpleStatementHandler extends BaseStatementHandler {

 @Override
  public void parameterize(Statement statement) {
    // N/A
  }

  }

PreparedStatementHandlerparameterize方法最终通过ParameterHandler接口经过多级中转后调用了PreparedStatement类中的参数赋值方法。

public class PreparedStatementHandler extends BaseStatementHandler {
  @Override
  public void parameterize(Statement statement) throws SQLException {
    parameterHandler.setParameters((PreparedStatement) statement);
  }
}

CallableStatementHandler中parameterize主要是通过registerOutputParameters方法中转后调用CallableStatement中的输出参数注册方法完成输出参数的注册,然后通过ParameterHandler接口经过多级中转后调用了PreparedStatement类中的参数赋值方法。

public class CallableStatementHandler extends BaseStatementHandler {
 /**
   * 对语句进行参数处理
   * @param statement SQL语句
   * @throws SQLException
   */
  @Override
  public void parameterize(Statement statement) throws SQLException {
    // 输出参数的注册
    registerOutputParameters((CallableStatement) statement);
    // 输入参数的处理
    parameterHandler.setParameters((CallableStatement) statement);
  }

}

到此这篇关于  mybatis源码解读之executor包语句处理功能的文章就介绍到这了,更多相关executor包语句处理内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

Java/Android 相关文章推荐
JPA如何使用entityManager执行SQL并指定返回类型
Jun 15 Java/Android
SpringCloud Alibaba项目实战之nacos-server服务搭建过程
Jun 21 Java/Android
解决SpringBoot跨域的三种方式
Jun 26 Java/Android
Java 数组内置函数toArray详解
Jun 28 Java/Android
SpringBoot读取Resource下文件的4种方法
Jul 02 Java/Android
SpringBoot工程下使用OpenFeign的坑及解决
Jul 02 Java/Android
SpringMVC 整合SSM框架详解
Aug 30 Java/Android
详解Java七大阻塞队列之SynchronousQueue
Sep 04 Java/Android
Java中的随机数Random
Mar 17 Java/Android
RestTemplate如何通过HTTP Basic Auth认证示例说明
Mar 17 Java/Android
SpringCloud Feign请求头删除修改的操作代码
Mar 20 Java/Android
Spring Cloud Netflix 套件中的负载均衡组件 Ribbon
Apr 13 Java/Android
java executor包参数处理功能 
Feb 15 #Java/Android
Java如何实现通过键盘输入一个数组
Feb 15 #Java/Android
Java实现给Word文件添加文字水印
Feb 15 #Java/Android
SSM项目使用拦截器实现登录验证功能
Jan 22 #Java/Android
利用Sharding-Jdbc进行分库分表的操作代码
关于MybatisPlus配置双数据库驱动连接数据库问题
Jan 22 #Java/Android
JavaCV实现照片马赛克效果
Jan 22 #Java/Android
You might like
require(),include(),require_once()和include_once()区别
2008/03/27 PHP
PHP 中执行排序与 MySQL 中排序
2009/04/21 PHP
php 魔术函数使用说明
2010/02/21 PHP
PHP 图片上传实现代码 带详细注释
2010/04/29 PHP
PHP实现全角字符转为半角方法汇总
2015/07/09 PHP
基于PHPexecl类生成复杂的报表表头示例
2016/10/14 PHP
javascript编程起步(第一课)
2007/01/10 Javascript
基于jQuery的遍历同id元素 并响应事件的代码
2012/06/14 Javascript
js几秒以后倒计时跳转示例
2013/12/26 Javascript
JavaScript判断前缀、后缀是否是空格的方法
2015/04/15 Javascript
详解XMLHttpRequest(二)响应属性、二进制数据、监测上传下载进度
2016/09/14 Javascript
bootstrap weebox 支持ajax的模态弹出框
2017/02/23 Javascript
jQuery动态产生select option下拉列表
2017/03/15 Javascript
vue学习笔记之vue1.0和vue2.0的区别介绍
2017/05/17 Javascript
Vue数据监听方法watch的使用
2018/03/28 Javascript
详解vue2.0监听属性的使用心得及搭配计算属性的使用
2018/07/18 Javascript
利用d3.js力导布局绘制资源拓扑图实例教程
2019/01/08 Javascript
微信小程序拼接图片链接无底洞深入探究
2019/09/03 Javascript
解决Vue-Router升级导致的Uncaught (in promise)问题
2020/08/07 Javascript
谈谈JavaScript中的函数
2020/09/08 Javascript
原生js实现简单轮播图
2020/10/26 Javascript
[02:05]2014DOTA2西雅图邀请赛 老队长全明星大猜想谁不服就按进显示器
2014/07/08 DOTA
[57:41]Secret vs Serenity 2018国际邀请赛小组赛BO2 第一场 8.16
2018/08/17 DOTA
浅析PHP与Python进行数据交互
2018/05/15 Python
Python jieba库用法及实例解析
2019/11/04 Python
jupyter notebook中美观显示矩阵实例
2020/04/17 Python
matlab、python中矩阵的互相导入导出方式
2020/06/01 Python
Python 实现RSA加解密文本文件
2020/12/30 Python
Python中使用Selenium环境安装的方法步骤
2021/02/22 Python
世界上最大的汽车共享网站:Zipcar
2017/01/14 全球购物
出门问问全球官方商城:Tichome音箱和TicWatch智能手表
2017/12/02 全球购物
金士达面试非笔试
2012/03/14 面试题
临床医学专业毕业生的自我评价
2013/10/17 职场文书
会计系中文个人求职信
2013/12/24 职场文书
教师个人成长总结
2015/02/11 职场文书
年底个人总结范文
2015/03/10 职场文书