返回顶部
首页 > 资讯 > 后端开发 > Python >Mybatis如何通过接口实现sql执行原理解析
  • 980
分享到

Mybatis如何通过接口实现sql执行原理解析

mybatis sql执行mybatis接口 2023-01-28 06:01:19 980人浏览 八月长安

Python 官方文档:入门教程 => 点击学习

摘要

目录1、环境搭建2、动态代理类的生成3、MapperProxy 增强 mapper 接口3.1、cachedInvoker(method)3.2、MapperMethod3.2.1、

使用过 mybatis 框架的小伙伴们都知道,mybatis 是个半 ORM 框架,通过写 mapper 接口就能自动实现数据库的增删改查,但是对其中的原理一知半解,接下来就让我们深入框架的底层一探究竟

1、环境搭建

首先引入 mybatis 的依赖,在 resources 目录下创建 mybatis 核心配置文件 mybatis-config.xml

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE configuration
        PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
        "https://mybatis.org/dtd/mybatis-3-config.dtd">
<configuration>

    <!-- 环境、事务工厂、数据源 -->
    <environments default="dev">
        <environment id="dev">
            <transactionManager type="JDBC"/>
            <dataSource type="UNPOOLED">
                <property name="driver" value="org.apache.derby.jdbc.EmbeddedDriver"/>
                <property name="url" value="jdbc:derby:db-user;create=true"/>
            </dataSource>
        </environment>
    </environments>

    <!-- 指定 mapper 接口-->
    <mappers>
        <mapper class="com.myboy.demo.mapper.user.UserMapper"/>
    </mappers>

</configuration>

在 com.myboy.demo.mapper.user 包下新建一个接口 UserMapper

public interface UserMapper {
    UserEntity getById(Long id);
    void insertOne(@Param("id") Long id, @Param("name") String name, @Param("JSON") List<String> json);
}

在 resources 的 com.myboy.demo.mapper.user 包下创建 UserMapper.xml

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper
        PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
        "Https://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.myboy.demo.mapper.user.UserMapper">

    <select id="getById" resultType="com.myboy.demo.db.entity.UserEntity">
        select * from demo_user where id = #{id}
    </select>

    <insert id="insertOne">
        insert into demo_user (id, name, json) values (#{id}, #{name}, #{json})
    </insert>
</mapper>

创建 main 方法测试

try(InputStream in = Resources.getResourceAsStream("com/myboy/demo/sqlsession/mybatis-config.xml")){
    SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(in);
    sqlSession = sqlSessionFactory.openSession();
    # 拿到代理类对象
    UserMapper mapper = sqlSession.getMapper(UserMapper.class);
    # 执行方法
    UserEntity userEntity = mapper.getById(2L);
    System.out.println(userEntity);
    sqlSession.close();
}catch (Exception e){
    e.printStackTrace();
}

2、动态代理类的生成

? 通过上面的示例,我们需要思考两个问题:

  • mybatis 如何生成 mapper 的动态代理类?
  • 通过 sqlSession.getMapper 获取到的动态代理类是什么内容?

通过查看源码,sqlSession.getMapper() 底层调用的是 mapperReGIStry 的 getMapper 方法

public <T> T getMapper(Class<T> type, SqlSession sqlSession) {
    // sqlSessionFactory build 的时候,就已经扫描了所有的 mapper 接口,并生成了一个 MapperProxyFactory 对象
    // 这里根据 mapper 接口类获取 MapperProxyFactory 对象,这个对象可以用于生成 mapper 的代理对象
    final MapperProxyFactory<T> mapperProxyFactory = (MapperProxyFactory<T>) knownMappers.get(type);
    if (mapperProxyFactory == null) {
      throw new BindingException("Type " + type + " is not known to the MapperRegistry.");
    }
    try {
      // 创建代理对象
      return mapperProxyFactory.newInstance(sqlSession);
    } catch (Exception e) {
      throw new BindingException("Error getting mapper instance. Cause: " + e, e);
    }
  }

代码注释已经写的很清楚,每个 mapper 接口在解析时会对应生成一个 MapperProxyFactory,保存到 knownMappers 中,mapper 接口的实现类(也就是动态代理类)通过这个 MapperProxyFactory 生成,mapperProxyFactory.newInstance(sqlSession) 代码如下:


public T newInstance(SqlSession sqlSession) {
    // 创建 MapperProxy 对象,这个对象实现 InvocationHandler 接口,里面封装类 mapper 动态代理方法的执行的核心逻辑
    final MapperProxy<T> mapperProxy = new MapperProxy<>(sqlSession, mapperInterface, methodCache);
    return newInstance(mapperProxy);
}

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

代码一目了然,通过 jdk 动态代理技术创建了 mapper 接口的代理对象,其 InvocationHandler 的实现是 MapperProxy,那么 mapper 接口中方法的执行,最终都会被 MapperProxy 增强

3、MapperProxy 增强 mapper 接口

MapperProxy 类实现了 InvocationHandler 接口,那么其核心方法必然是在其 invoke 方法内部


@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
    try {
      // 如果执行的方法是 Object 类的方法,则直接反射执行
      if (Object.class.equals(method.getDeclarinGClass())) {
        return method.invoke(this, args);
      } else {
        // 1、根据method创建方法执行器对象 MapperMethodInvoker,用于适配不同的方法执行过程
        // 2、执行方法逻辑
        return cachedInvoker(method).invoke(proxy, method, args, sqlSession);
      }
    } catch (Throwable t) {
      throw ExceptionUtil.unwrapThrowable(t);
    }
}

3.1、cachedInvoker(method)

由于 jdk8 对接口增加了 default 关键字,使接口中的方法也可以有方法体,但是默认方法和普通方法的反射执行方式不同,需要用适配器适配一下才能统一执行,具体代码如下


private MapperMethodInvoker cachedInvoker(Method method) throws Throwable {
    try {
      return MapUtil.computeIfAbsent(methodCache, method, m -> {
        // 返回默认方法执行器 DefaultMethodInvoker
        if (m.isDefault()) {
          try {
            if (privateLookupInMethod == null) {
              return new DefaultMethodInvoker(getMethodHandleJava8(method));
            } else {
              return new DefaultMethodInvoker(getMethodHandleJava9(method));
            }
          } catch (IllegalAccessException | InstantiationException | InvocationTargetException
              | NoSuchMethodException e) {
            throw new RuntimeException(e);
          }
        }
        // 返回普通方法执行器,只有一个 invoke 执行方法,实际上就是调用 MapperMethod 的执行方法
        else {
          return new PlainMethodInvoker(new MapperMethod(mapperInterface, method, sqlSession.getConfiguration()));
        }
      });
    } catch (RuntimeException re) {
      Throwable cause = re.getCause();
      throw cause == null ? re : cause;
    }
}

如果判定执行的是接口的默认方法,则原始方法封装成 DefaultMethodInvoker,这个类的 invoke 方法就是利用反射调用原始方法,没什么好说的

如果是普通的接口方法,则将方法封装成封装成 MapperMethod,然后再将 MapperMethod 封装到 PlainMethodInvoker 中,PlainMethodInvoker 没什么好看的,底层的执行方法还是调用 MapperMethod 的执行方法,至于 MapperMethod,咱们放到下一章来看

3.2、MapperMethod

首先看下构造方法

public MapperMethod(Class<?> mapperInterface, Method method, Configuration config) {
    // 通过这个 SqlCommand 可以拿到 sql 类型和sql 对应的 MappedStatement
    this.command = new SqlCommand(config, mapperInterface, method);
    // 包装了 mapper 接口的一个方法,可以拿到方法的信息,比如方法返回值类型、返回是否集合、返回是否为空
    this.method = new MethodSignature(config, mapperInterface, method);
}

代码里的注释写的很清楚了,MapperMethod 构造方法创建了两个对象 SqlCommand 和 MethodSignature

mapper 接口的执行核心逻辑在其 execute() 方法中:


  public Object execute(SqlSession sqlSession, Object[] args) {
    Object result;
    switch (command.getType()) {
      case INSERT: {
        // 参数处理,单个参数直接返回,多个参数封装成 map
        Object param = method.convertArgsToSqlCommandParam(args);
        // 调用 sqlSession 的插入方法
        result = rowCountResult(sqlSession.insert(command.getName(), param));
        break;
      }
      case UPDATE: {
        Object param = method.convertArgsToSqlCommandParam(args);
        result = rowCountResult(sqlSession.update(command.getName(), param));
        break;
      }
      case DELETE: {
        Object param = method.convertArgsToSqlCommandParam(args);
        result = rowCountResult(sqlSession.delete(command.getName(), param));
        break;
      }
      case SELECT:
        if (method.returnsVoid() && method.hasResultHandler()) {
          // 方法返回值为 void,但是参数里有 ResultHandler
          executeWithResultHandler(sqlSession, args);
          result = null;
        } else if (method.returnsMany()) {
          // 方法返回集合
          result = executeForMany(sqlSession, args);
        } else if (method.returnsMap()) {
          // 方法返回 map
          result = executeForMap(sqlSession, args);
        } else if (method.returnsCursor()) {
          // 方法返回指针
          result = executeForCursor(sqlSession, args);
        } else {
          // 方法返回单个对象
          // 将参数进行转换,如果是一个参数,则原样返回,如果多个参数,则返回map,key是参数name(@Param注解指定 或 arg0、arg1 或 param1、param2 ),value 是参数值
          Object param = method.convertArgsToSqlCommandParam(args);
          // selectOne 从数据库获取数据,封装成返回值类型,取出第一个
          result = sqlSession.selectOne(command.getName(), param);

          // 如果返回值为空,并且返回值类型是 Optional,则将返回值用 Optional.ofNullable 包装
          if (method.returnsOptional()
              && (result == null || !method.getReturnType().equals(result.getClass()))) {
            result = Optional.ofNullable(result);
          }
        }
        break;
      case FLUSH:
        result = sqlSession.flushStatements();
        break;
      default:
        throw new BindingException("Unknown execution method for: " + command.getName());
    }
    if (result == null && method.getReturnType().isPrimitive() && !method.returnsVoid()) {
      throw new BindingException("Mapper method '" + command.getName()
          + " attempted to return null from a method with a primitive return type (" + method.getReturnType() + ").");
    }
    return result;
  }

代码逻辑很清晰,拿 Insert 方法来看,他只做了两件事

  • 参数转换
  • 调用 sqlSession 对应的 insert 方法

3.2.1、参数转换 method.convertArgsToSqlCommandParam(args)

在 mapper 接口中,假设我们定义了一个 user 的查询方法

List<User> find(@Param("name")String name, @Param("age")Integer age)

在我们的 mapper.xml 中,写出来的 sql 可以是这样的:

select * from user where name = #{name} and age > #{age}

当然不使用 @Param 注解也可以的,按参数顺序来

select * from user where name = #{arg0} and age > #{arg1}
或
select * from user where name = #{param1} and age > #{param2}

因此如果要通过占位符匹配到具体参数,就要将接口参数封装成 map 了,如下所示

{arg1=12, arg0="abc", param1="abc", param2=12}
或
{name="abc", age=12, param1="abc", param2=12}
复制代码

这里的这个 method.convertArgsToSqlCommandParam(args) 就是这个作用,当然只有一个参数的话就不用转成 map 了, 直接就能匹配

3.2.2、调用 sqlSession 的方法获取结果

真正要操作数据库还是要借助 sqlSession,因此很快就看到了 sqlSession.insert(command.getName(), param) 方法的执行,其第一个参数是 statement 的 id,就是 mpper.xml 中 namespace 和 insert 标签的 id的组合,如 com.myboy.demo.mapper.MoonAppMapper.getAppById,第二个参数就是上面转换过的参数,至于 sqlSession 内部处理逻辑,不在本章叙述范畴

sqlSession 方法执行完后的执行结果交给 rowCountResult 方法处理,这个方法很简单,就是将数据库返回的数据处理成接口返回类型,代码很简单,如下

  private Object rowCountResult(int rowCount) {
    final Object result;
    if (method.returnsVoid()) {
      result = null;
    } else if (Integer.class.equals(method.getReturnType()) || Integer.TYPE.equals(method.getReturnType())) {
      result = rowCount;
    } else if (Long.class.equals(method.getReturnType()) || Long.TYPE.equals(method.getReturnType())) {
      result = (long) rowCount;
    } else if (Boolean.class.equals(method.getReturnType()) || Boolean.TYPE.equals(method.getReturnType())) {
      result = rowCount > 0;
    } else {
      throw new BindingException("Mapper method '" + command.getName() + "' has an unsupported return type: " + method.getReturnType());
    }
    return result;
  }

4、小结

到目前为止,我们已经搞清楚了通过 mapper 接口生成动态代理对象,以及代理对象调用 sqlSession 操作数据库的逻辑,我总结出执行逻辑图如下:

总结

到此这篇关于Mybatis如何通过接口实现sql执行原理的文章就介绍到这了,更多相关Mybatis接口实现sql执行内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

--结束END--

本文标题: Mybatis如何通过接口实现sql执行原理解析

本文链接: https://lsjlt.com/news/178340.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • Mybatis如何通过接口实现sql执行原理解析
    目录1、环境搭建2、动态代理类的生成3、MapperProxy 增强 mapper 接口3.1、cachedInvoker(method)3.2、MapperMethod3.2.1、...
    99+
    2023-01-28
    mybatis sql执行 mybatis接口
  • SQL语句解析执行的过程及原理
    目录一、sqlSession简单介绍二、获得sqlSession对象源码分析三、SQL执行流程,以查询为例一、sqlSession简单介绍 拿到SqlSessionFactory对象...
    99+
    2024-04-02
  • MyBatis接口的简单实现原理分析
    用过MyBatis3的人可能会觉得为什么MyBatis的Mapper接口没有实现类,但是可以直接用?那是因为MyBatis使用Java动态代理实现的接口。这里仅仅举个简单例子来说明原理,不是完全针对MyBatis的,这种思想我们也可以应用在...
    99+
    2023-05-31
    mybatis 接口 原理
  • MySQL中通过EXPLAIN如何分析SQL的执行计划详解
    前言 在MySQL中,我们可以通过EXPLAIN命令获取MySQL如何执行SELECT语句的信息,包括在SELECT语句执行过程中表如何连接和连接的顺序。 下面分别对EXPLAIN命令结果的每一列...
    99+
    2024-04-02
  • 详解Springboot如何通过注解实现接口防刷
    目录前言1、实现防刷切面PreventAop.java1.1 定义注解Prevent1.2 实现防刷切面PreventAop2、使用防刷切面3、演示前言 本文介绍一种极简洁、灵活通用...
    99+
    2024-04-02
  • 如何理解PHP程序执行的过程原理
    目录概述加载php.ini配置加载php内置扩展概述 Web环境我们假设为Apache。在编译PHP的时候,为了能够让Apache支持PHP,我们会生成一个mod_php5.so的模...
    99+
    2024-04-02
  • MyBatis详解如何实现Dao层接口
    目录传统开发方式编写UserDao接口编写UserDaompl实现传统测试方法代理开发方法代理开发方式介绍编写UserMapper接口测试代理方法传统开发方式 编写UserDao接口...
    99+
    2024-04-02
  • java应用开发之Mybatis通过Mapper代理自定义接口的实现
    如何实现?主要分为以下两步骤 1.通过 Mapper 代理实现⾃定义接口 2.编写与方法相对应的 Mapper.xml 1.自定义接口Account...
    99+
    2024-04-02
  • 如何通过ASP接口实现高效的自然语言处理?
    随着人工智能技术的不断发展,自然语言处理已经成为了人工智能领域中的重要分支。而ASP接口,作为一种常用的编程语言,也可以用于实现高效的自然语言处理。在本文中,我们将会介绍如何通过ASP接口实现高效的自然语言处理,并提供一些实用的代码演示。 ...
    99+
    2023-10-04
    接口 自然语言处理 数据类型
  • Java 如何通过注解实现接口输出时数据脱敏
    目录Java注解实现接口输出数据脱敏先声明了一个注解我们目前只支持对手机号然后我们需要实现注解的拦截功能我对默认声明和脱敏名称和手机号进行了测试Java注解的字段脱敏处理定义需要脱敏...
    99+
    2024-04-02
  • 如何通过一个注解实现MyBatis字段加解密
    目录简介模块使用方法配置项说明开源链接总结简介 mybatis-crypto 是一个基于 mybatis 插件机制实现的字段加解密组件,通过一个注解即可对敏感数据进行加解密处理。 支...
    99+
    2024-04-02
  • 详解Java是如何通过接口来创建代理并进行http请求
    场景 现在想要做这么一个事情,公司的dubbo服务都是内网的,但是提供了一个对外的出口,通过链接就能请求到对应的dubbo服务。(具体怎么做的应该就是个网关,然后将http请求转为d...
    99+
    2024-04-02
  • 利用Java如何获取Mybatis动态生成的sql接口实现
    目录前言1、编写xml: SqlGenarate.mapper.xml2、定义接口3、实现接口总结前言 如果你有使用 JDBC 或其他类似框架的经验,你就能体会到根据不同条件拼接 S...
    99+
    2024-04-02
  • 如何通过PHP实现HTTP接口索引的高效访问?
    PHP是一种流行的服务器端编程语言,常用于Web应用程序的开发。在Web开发中,HTTP接口索引是一个非常重要的组件。它可以让用户更加方便地查找和访问Web API,提高Web应用的性能和用户体验。本文将介绍如何通过PHP实现HTTP接口索...
    99+
    2023-08-12
    http 接口 索引
  • 如何通过Python调用接口实现抠图并改底色
    这篇文章主要介绍了如何通过Python调用接口实现抠图并改底色的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇如何通过Python调用接口实现抠图并改底色文章都会有所收获,下面我们一起来看看吧。一、注册百度AI账...
    99+
    2023-07-04
  • 如何通过Java接口来实现大数据处理?学习笔记分享
    随着互联网和物联网技术的快速发展,数据量的爆发式增长已经成为一种趋势。如何高效地处理这些大数据成为了各个行业所面临的一个共同难题。在这种背景下,Java作为一种广泛使用的编程语言,也在大数据处理方面发挥着越来越重要的作用。本文将介绍如何通...
    99+
    2023-10-17
    大数据 接口 学习笔记
  • 如何通过SQL语句在MongoDB中实现事务处理?
    如何通过SQL语句在MongoDB中实现事务处理?摘要:作为一种非关系型数据库,MongoDB一直以其高性能和可扩展性而闻名。然而,对于需要进行事务处理的应用程序而言,MongoDB在较早的版本中并不支持事务功能。不过,从MongoDB 4...
    99+
    2023-12-17
    MongoDB 事务处理 SQL语句
  • 如何通过ECharts和php接口实现实时统计图的展示
    如何通过ECharts和PHP接口实现实时统计图的展示随着互联网和大数据技术的快速发展,数据可视化成为了重要的一环。而ECharts作为一款优秀的开源JavaScript数据可视化库,能够帮助我们简单、高效地实现各种统计图的展示。本文将介绍...
    99+
    2023-12-17
    echarts PHP接口 实时统计图
  • 如何进行Spring源码剖析AOP实现原理
    今天就跟大家聊聊有关如何进行Spring源码剖析AOP实现原理,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。前言前面写了六篇文章详细地分析了Spring Bean加载流程,这部分完了...
    99+
    2023-06-02
  • 使用mybatis执行SQL语句时有参数出现返回NULL值如何解决
    今天就跟大家聊聊有关使用mybatis执行SQL语句时有参数出现返回NULL值如何解决,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。原来的写法:<select id=...
    99+
    2023-05-31
    mybatis sql null
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作