`
情情说
  • 浏览: 39081 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

深入浅出MyBatis:MyBatis解析和运行原理

 
阅读更多

 

上一篇介绍了反射和动态代理基础,主要是为本篇文章做个铺垫,反射使配置和灵活性大大提高,可以给很多配置设置参数,动态代理可以在运行时创建代理对象,做一些特殊的处理。

本篇会介绍MyBatis解析和运行原理,下一篇介绍插件及应用,目的是更好地编写插件,通过本篇的介绍,你会了解到:

  • 构建SqlSessionFactory过程
  • 映射器的动态代理
  • SqlSession的4大对象
  • sql执行的过程

SqlSessionFactory和SqlSession是MyBatis的核心组件,在文章 JDBC和MyBatis介绍 中有详细说明。

构建SqlSessionFactory过程

构建主要分为2步:

  • 通过XMLConfigBuilder解析配置的XML文件,读出配置参数,包括基础配置XML文件和映射器XML文件;
  • 使用Configuration对象创建SqlSessionFactory,SqlSessionFactory是一个接口,提供了一个默认的实现类DefaultSqlSessionFactory。

说白了,就是将我们的所有配置解析为Configuration对象,在整个生命周期内,可以通过该对象获取需要的配置。

由于插件需要频繁访问映射器的内部组成,会重点这部分,了解这块配置抽象出来的对象:

MappedStatement

它保存映射器的一个节点(select|insert|delete|update),包括配置的SQL,SQL的id、缓存信息、resultMap、parameterType、resultType等重要配置内容。

它涉及的对象比较多,一般不去修改它。

SqlSource

它是MappedStatement的一个属性,主要作用是根据参数和其他规则组装SQL,也是很复杂的,一般也不用修改它。

BoundSql

对于参数和SQL,主要反映在BoundSql类对象上,在插件中,通过它获取到当前运行的SQL和参数以及参数规则,作出适当的修改,满足特殊的要求。

BoundSql提供3个主要的属性:parameterObject、parameterMappings和sql,下面分别来介绍。

parameterObject为参数本身,可以传递简单对象、POJO、Map或@Param注解的参数:

  • 传递简单对象(int、float、String等),会把参数转换为对应的类,比如int会转换为Integer;
  • 如果传递的是POJO或Map,paramterObject就是传入的POJO或Map不变;
  • 如果传递多个参数,没有@Param注解,parameterObject就是一个Map<string,object>对象,类似这样的形式{“1”:p1 , “2”:p2 , “3”:p3 … “param1”:p1 , “param2”:p2 , “param3”,p3 …},所以在编写的时候可以使用#{param1}或#{1}去引用第一个参数;
  • 如果传递多个参数,有@Param注解,与没有注解的类似,只是将序号的key替换为@Param指定的name;

parameterMappings,它是一个List,元素是ParameterMapping对象,这个对象会描绘sql中的参数引用,包括名称、表达式、javaType、jdbcType、typeHandler等信息。

sql,是写在映射器里面的一条sql。

有了Configuration对象,构建SqlSessionFactory就简单了:

sqlSessionFactory = new SqlSessionFactoryBuilder().bulid(inputStream);

SqlSession运行过程

映射器的动态代理

Mapper映射是通过动态代理来实现的,使用JDK动态代理返回一个代理对象,供调用者访问。

首先看看实现InvocationHandler接口的类,它是执行本代理方法的关键,可以看到,Mapper是一个接口,会生成MapperMethod对象,调用execute方法。

public class MapperProxy<T> implements InvocationHandler, Serializable {

  .....

  @Override
  public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
    try {
      if (Object.class.equals(method.getDeclaringClass())) {
        return method.invoke(this, args);
      } else if (isDefaultMethod(method)) {
        return invokeDefaultMethod(proxy, method, args);
      }
    } catch (Throwable t) {
      throw ExceptionUtil.unwrapThrowable(t);
    }
    final MapperMethod mapperMethod = cachedMapperMethod(method);
    return mapperMethod.execute(sqlSession, args);
  }
}

看下面的代码,MapperMethod采用命令模式,根据不同的sql操作,做不同的处理。

public class MapperMethod {
  public Object execute(SqlSession sqlSession, Object[] args) {
    Object result;
    switch (command.getType()) {
      case INSERT: {
      Object param = method.convertArgsToSqlCommandParam(args);
        result = rowCountResult(sqlSession.insert(command.getName(), param));
        break;
      }
      case UPDATE: {
        Object param = method.convertArgsToSqlCommandParam(args);
        result = rowCountResult(sqlSession.update(command.getName(), param));
        break;

        ......

      }
    }
  }

最后看下,生成代理类的方法,就是使用JDK动态代理Proxy来创建的。

public class MapperProxyFactory<T> {

  public T newInstance(SqlSession sqlSession) {
    final MapperProxy<T> mapperProxy = new MapperProxy<T>(sqlSession, mapperInterface, methodCache);
    return newInstance(mapperProxy);
  }

  @SuppressWarnings("unchecked")
  protected T newInstance(MapperProxy<T> mapperProxy) {
    return (T) Proxy.newProxyInstance(mapperInterface.getClassLoader(), new Class[] { mapperInterface }, mapperProxy);
  }

}

总结下映射器的调用过程,返回的Mapper对象是代理对象,当调用它的某个方法时,其实是调用MapperProxy#invoke方法,而映射器的XML文件的命名空间对应的就是这个接口的全路径,会根据全路径和方法名,便能够绑定起来,定位到sql,最后会使用SqlSession接口的方法使它能够执行查询。

SqlSession下的四大对象

通过上面的分析,映射器就是一个动态代理对象,进入到了MapperMethod的execute方法,它经过简单的判断就进入了SqlSession的删除、更新、插入、选择等方法,这些方法如何执行是下面要介绍的内容。

Mapper执行的过程是通过Executor、StatementHandler、ParameterHandler和ResultHandler来完成数据库操作和结果返回的,理解他们是编写插件的关键:

  • Executor:执行器,由它统一调度其他三个对象来执行对应的SQL;
  • StatementHandler:使用数据库的Statement执行操作;
  • ParameterHandler:用于SQL对参数的处理;
  • ResultHandler:进行最后数据集的封装返回处理;

在MyBatis中存在三种执行器:

  • SIMPLE:简易执行器,默认的执行器;
  • REUSE:执行重用预处理语句;
  • BATCH:执行重用语句和批量更新,针对批量专用的执行器;

以SimpleExecutor为例,说明执行过程

public class SimpleExecutor extends BaseExecutor {

  /**
  * 执行查询操作
  */
  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(this, ms, parameter, rowBounds, resultHandler, boundSql);
      stmt = prepareStatement(handler, ms.getStatementLog());
      return handler.<E>query(stmt, resultHandler);
    } finally {
      closeStatement(stmt);
    }
  }

  /**
  * 初始化StatementHandler
  */
  private Statement prepareStatement(StatementHandler handler, Log statementLog) throws SQLException {
    Statement stmt;
    Connection connection = getConnection(statementLog);
    stmt = handler.prepare(connection);
    handler.parameterize(stmt);
    return stmt;
  }

  /**
  * 执行查询
  */
  @Override
  public <E> List<E> query(Statement statement, ResultHandler resultHandler) throws SQLException {
    String sql = boundSql.getSql();
    statement.execute(sql);
    return resultSetHandler.<E>handleResultSets(statement);
  }
}

可以看到最后会委托给StatementHandler会话器进行处理,它是一个接口,实际创建的是RoutingStatementHandler对象,但它不是真实的服务对象,它是通过适配器模式找到对应的StatementHandler执行的。在MyBatis中,StatementHandler和Executor一样分为三种:SimpleStatementHandler、PreparedStatementHandler、CallableStatementHandler。

Executor会先调用StatementHandler的prepare方法预编译SQL语句,同时设置一些基本运行的参数。然后调用parameterize()方法启用ParameterHandler设置参数,完成预编译,跟着执行查询,用ResultHandler封装结果返回给调用者。

参数处理器和结果处理器比较简单,就不在此介绍了。

下一篇会介绍插件及其应用,主要是在sql执行的过程中,在四大对象的基础上进行扩展。

欢迎扫描下方二维码,关注我的个人微信公众号 ~

情情说

 

0
0
分享到:
评论

相关推荐

    深入浅出MyBatis技术原理与实战

    《深入浅出MyBatis技术原理与实战》这本书旨在帮助读者全面理解MyBatis这一流行的Java持久层框架,深入解析其工作原理,并提供实战指导。MyBatis是一个轻量级的ORM(对象关系映射)框架,它允许开发者将SQL语句与...

    深入浅出MyBatis技术原理与实践

    《深入浅出MyBatis技术原理与实践》是一本专注于解析MyBatis这一流行Java持久层框架的专业书籍。MyBatis作为一个轻量级的ORM(对象关系映射)框架,它将SQL语句与Java代码分离,使得开发更加灵活且易于维护。这本书...

    腾讯T4架构师耗时八个月整理《深入浅出MyBatis技术原理与实战》

    《深入浅出MyBatis技术原理与实战》是由腾讯T4架构师精心整理的一本关于MyBatis框架的专业书籍,旨在帮助从初级到中高级的开发人员深入理解MyBatis的使用和内在机制。MyBatis作为一个优秀的持久层框架,其核心特性...

    Spring技术内幕:深入解析Spring架构与设计原理.rar

    总之,《Spring技术内幕:深入解析Spring架构与设计原理》是一本深入浅出的Spring指南,涵盖了从基础到高级的各个层面,无论你是初学者还是经验丰富的开发者,都能从中受益匪浅,提升对Spring框架的理解和应用能力。...

    贯穿案例:超市订单管理系统

    【描述】:北大青鸟作为知名的IT教育机构,其教学案例通常深入浅出,注重理论与实践相结合。在这个案例中,学员将学习到如何设计并实现一个基于Java技术栈的后台管理系统。系统不仅涉及数据库设计,还包含了完整的源...

    深入Spring2-轻量级J2EE开发框架原理与实践

    总之,《深入Spring2:轻量级J2EE开发框架原理与实践》是一本深入浅出的Spring学习资料,适合希望掌握Spring框架的开发者阅读。通过这本书,读者不仅可以了解Spring2的核心理念,还能学习到如何将这些理念转化为实践...

    Java 208道面试.docx

    - Spring MVC的工作原理,控制器、模型、视图和视图解析器的角色。 11. **Spring Boot/Spring Cloud**: - Spring Boot的自动配置和起步依赖,简化项目配置。 - Spring Cloud的微服务治理,如Eureka、Ribbon、...

    SpringMVC 名片管理系统

    本资源通过一个名片管理系统的实例,深入浅出地介绍了SpringMVC的工作原理以及如何与数据库进行交互,旨在帮助初学者理解和掌握这一强大的技术。 首先,SpringMVC是Spring框架的一个模块,它主要负责处理HTTP请求,...

    Java耿祥义主编课件

    Java耿祥义主编的课件深入浅出地讲解了Java的基础知识,包括: 1. **环境搭建**:介绍如何下载和安装Java开发工具包(JDK),设置环境变量,以及如何使用Java编译器javac和解释器java。 2. **语法基础**:讲解Java...

    韩顺平java笔记完整版

    【描述】:这份笔记是韩顺平老师教学经验的结晶,旨在帮助学习者系统地理解和掌握Java语言,通过深入浅出的方式讲解了Java的核心概念、语法特性以及实际开发中的应用。 【标签】: 1. **韩顺平**:中国知名IT教育家...

    Java开发技术大全 清晰版

    《Java开发技术大全》是一...这本书深入浅出地讲解了Java开发的各个方面,无论是初学者还是有经验的开发者,都能从中受益。通过学习,读者将能够熟练运用Java进行软件开发,并为进阶到更高级的技术领域打下坚实的基础。

    Pro Spring 3

    《Pro Spring 3》不仅是一本技术手册,更是一部深入浅出的学习指南。它不仅涵盖了Spring框架的所有核心特性和功能,还提供了丰富的实践案例和最佳实践建议,非常适合希望深入了解Spring框架并应用于实际项目中的...

    spring boot2.x 书籍

    《深入浅出Spring Boot 2.x》这本书是针对Spring Boot 2.x版本的全面解析,旨在帮助开发者深入了解和熟练掌握这一流行的Java开发框架。Spring Boot以其便捷的起步配置、自动配置特性以及开箱即用的功能,极大地简化...

    java开发实战经典_MLDN李兴华 java源码和随着视频的pdf

    讲义中可能对诸如设计模式、JVM工作原理、垃圾回收机制、Spring框架、MyBatis等企业级开发中的常见技术有深入浅出的讲解。 学习这个压缩包,你需要先从基础开始,理解Java语言的基本结构和语法规则,然后逐步深入到...

    若依框架(RuoYi-Cloud 微服务版本)

    - **基础概念讲解**:深入浅出地解释Spring Boot、Spring Cloud Alibaba的基本原理和使用方法。 - **实战项目演示**:通过具体的项目案例,手把手指导如何使用若依框架构建完整的微服务应用。 - **高级特性探索**:...

    spring in action英文原版

    这本书深入浅出地介绍了Spring框架的核心概念和技术,帮助读者掌握如何在实际项目中有效地使用Spring。作为一本英文原版,它为中文读者提供了直接接触原汁原味的技术文献的机会,有助于提升对Spring框架的深入理解。...

Global site tag (gtag.js) - Google Analytics