首页 > 编程语言 >一次Mybaits查询的源码分析

一次Mybaits查询的源码分析

时间:2023-06-17 21:22:19浏览次数:42  
标签:xml mapper 调用 对象 查询 Mybaits 源码 对应

很好奇Mybaits是怎么将xml和mapper对应起来的,用一段比较简单的demo去debug追踪一下源码看看

先用xml配置的方式,看懂了再去看注解的方式是怎么实现的

获取Mapper

Mybaits是如何从xml中加载到mapper的

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE configuration
        PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
        "http://mybatis.org/dtd/mybatis-3-config.dtd">
<configuration>
    <typeAliases>
        <package name="com.github.yeecode.mybatisdemo"/>
    </typeAliases>
    <environments default="development">
        <environment id="development">
            <transactionManager type="JDBC"/>
                <dataSource type="POOLED">
                    <property name="driver" value="com.mysql.cj.jdbc.Driver"/>
                    <property name="url" value="jdbc:mysql://127.0.0.1:3306/test?serverTimezone=UTC"/>
                    <property name="username" value="root"/>
                    <property name="password" value="123456"/>
                </dataSource>
        </environment>
    </environments>
    <mappers>
        <mapper resource="com/github/yeecode/mybatisdemo/UserMapper.xml"/>
    </mappers>
</configuration>

在xml中有mapper标签,应该是从这里加载到配置

示例代码

  public static void main(String[] args) {
        // 第一阶段:MyBatis的初始化阶段
        String resource = "mybatis-config.xml";
        // 得到配置文件的输入流
        InputStream inputStream = null;
        try {
            inputStream = Resources.getResourceAsStream(resource);
        } catch (IOException e) {
            e.printStackTrace();
        }
        // 得到SqlSessionFactory
        SqlSessionFactory sqlSessionFactory =
                new SqlSessionFactoryBuilder().build(inputStream);

        // 第二阶段:数据读写阶段
        try (SqlSession session = sqlSessionFactory.openSession()) {
            // 找到接口对应的实现
            UserMapper userMapper = session.getMapper(UserMapper.class);
            // 组建查询参数
            User userParam = new User();
            userParam.setSchoolName("Sunny School");
            // 调用接口展开数据库操作
            List<User> userList =  userMapper.queryUserBySchoolName(userParam);
            // 打印查询结果
            for (User user : userList) {
                System.out.println("name : " + user.getName() + " ;  email : " + user.getEmail());
            }
        }
    }

UserMapper userMapper = session.getMapper(UserMapper.class);此处开始debug,看看是怎么获取到mapper的

一路点进来发现是从一个Map中去获取mapper对象

private final Map<Class<?>, MapperProxyFactory<?>> knownMappers = new HashMap<>();

那么是从什么时候填充knownMappers的呢

这个对象的方法不只有获取mapper,还有添加mapper,找一个添加方法继续断底点

找到是找到了,但是不知道什么时候调用的,可以通过IDEA的调用栈

最后发现了,是加载xml的时候,去解析mapper标签里的值,然后再通过类加载器去加载资源,最后加载到knownMappers中,还有去解析xml中的sql的过程

这样子xml就和mapper就对应起来了,虽然知道了mapper和xml的对应关系,但是不知道怎么通过调用mapper里的方法,去找到对应的sql

在对 List<User> userList = userMapper.queryUserBySchoolName(userParam);debug时,没有进入到mapper方法中,而是会进入到一个代理类中

刚刚在getMapper()中给UserMapper创建了代理,那么大概知道是mapper和xml是怎么关联的了,当调用mapper时,会被MapperProxy代理,去执行查询方法时,通过上边的knownMappers
获取到mapper对应的xml,这样代理类就知道要调用的方法和对应的sql的哪里

最终时通过mapperMethod.execute(sqlSession, args);去执行查询的,点进去一看发现对各种sql类型做了处理

select的查询原来通过返回值来选择不同的处理

很好奇这些属性是怎么判断的,找到对应的类继续断点

原来是在execute之前去赋值,而且这个方法会把调用方法对应的xml中的方法找到

通过获取到方法的返回值,然后再去做对比,我这个方法返回的是list,就命中了returnMany,在继续断点,找到了真正执行的方法

这里就已经将关联的xml信息带过来了


继续看会看到缓存相关的代码,如果命中了缓存就直接返回了,我这里没有就继续开了一个线程往下执行,delegate是一个Executor

最后的查询到了这里,就是调用mysql的包了,在statement中,已经把sql、参数和连接配置什么的都封装好了

查询完后把结果返回到statement,但是返回的内容很多,查询结果记录在这里

查询总结:

  • 在进行数据库查询前,先查询缓存;没有名中,则数据库查询之后的结果也放入缓存中
  • SQL 语句的执行经过了层层转化,依次经过了 MappedStatement 对象、Statement对象和 PreparedStatement对象,最后才交给mysql执行
  • 最终数据库查询得到的结果交给 ResultHandler对象处理

返回结果

将结果映射到实体类上这段代码有点绕,调用链很长

首先是这里先创建输出的实体类,就是resultMap里定义的对象

创建好实体后,把实体传输给下一个方法,填充实体

将实体字段和结果集里的字段对应起来,然后根据字段去获取对应的值,然后把值设置到实体里,通过循环遍历全部字段

这样走一圈回来,一个对象就映射好了,再经过循环,就把全部的对象都拿到了,最后再将这些对象封装到multipleResults集合里,这个集合就是返回值了

映射总结:

  • 获取并创建实体类
  • 将实体类的字段和结果集的字段一一对应,然后再填充实体的值
  • 最后返回实体集合

总结

以上就是Mybaits读取xml,然后查询的过程了,整个过程还是很复杂的,很多层封装和跳转,但是大大的提高了我们开发的效率

然后再把总结发一下

获取配置总结:

  • 得到配置文件然后转换成输入流
  • 将输入流传给SqlSessionFactoryBuilder创建SqlSessionFactory
  • 扫描xml文件并加载,然后将xml和mapper的对应关系填充好

查询总结:

  • 在进行数据库查询前,先查询缓存;没有名中,则数据库查询之后的结果也放入缓存中
  • SQL 语句的执行经过了层层转化,依次经过了 MappedStatement 对象、Statement对象和 PreparedStatement对象,最后才交给mysql执行
  • 最终数据库查询得到的结果交给 ResultHandler对象处理

映射总结:

  • 获取并创建实体类
  • 将实体类的字段和结果集的字段一一对应,然后再填充实体的值
  • 最后返回实体集合

标签:xml,mapper,调用,对象,查询,Mybaits,源码,对应
From: https://www.cnblogs.com/aruo/p/17486845.html

相关文章

  • 从源码级深入剖析Tomcat类加载原理
    众所周知,Java中默认的类加载器是以父子关系存在的,实现了双亲委派机制进行类的加载,在前文中,我们提到了,双亲委派机制的设计是为了保证类的唯一性,这意味着在同一个JVM中是不能加载相同类库的不同版本的类。然而与许多服务器应用程序一样,Tomcat允许容器的不同部分以及在容器上运行的......
  • 【sentinel】sentinel客户端与dashboard通讯源码分析
    Sentinel客户端的规则大部分都是在Dashboard上完成配置,那么Sentinel客户端与Dashboard之间是如何进行通讯的呢?客户端定时向dashboard发送心跳要想实现Sentinel客户端与Dashboard进行通讯,我们需要引入下面的依赖:<dependency><groupId>com.alibaba.csp</groupId><artifactI......
  • 【框架源码】Spring源码解析之Bean创建源码流程
    问题:Spring中是如何初始化单例bean的?我们都知道Spring解析xml文件描述成BeanDefinition,解析BeanDefinition最后创建Bean将Bean放入单例池中,那么Spring在创建Bean的这个过程都做了什么。Spring核心方法refresh()中最最重要的一个方法finishBeanFactoryInitialization()方法,该方法......
  • 源码泄露+bak备份泄露+vim泄露+.DS_Store(mas迁移泄露)
    源码泄露+bak备份泄露+vim泄露+.DS_Store(mas迁移泄露)1.源码泄露web网站源码打包在web目录下造成泄露,通常以压缩包方式存在,如.zip、.rar、.tar、.tar.gz等,常见命名方式为网站名,www.网站名,backup+网站名等简单入门题目扫描到压缩包文件进行下载,找到对应文件,查看是否有flag,如果没......
  • 开源数字药店系统源码:打造高效的医药销售平台
    作为医药销售的全新解决方案,数字药店系统源码能够为医药企业提供更高效的销售解决方案,提高企业的竞争力。本文将详细介绍开源数字药店系统源码的特点和优势,以及如何打造高效的医药销售平台。一、开源数字药店系统源码的特点1. 功能丰富具有完善的功能,包括商品管理、订单管理、客户......
  • .Net Core医学检验LIS系统源码
    .NetCoreLIS系统源码,在第三方快检实验室、二级医院检验科应用5年以上,系统运行稳定、功能齐全,界面布局合理、操作简便。系统采用B/S架构SaaS模式,可扩展性强。LIS系统为实验室服务对象提供检验申请、采集标本、结果查询等功能;为实验室工作人员的核收标本、分送标本、传送资料、分析......
  • 互联网医院成品|互联网医院源码|线上医疗所含功能
    作为医疗服务领域的新生力量,互联网医院系统逐渐成为了医疗行业内时常交流和沟通的名词和产品,互联网医院系统就是用信息化、互联网化的手段去降低成本、优化服务,提升医疗服务诊前、诊中、诊后的全流程质量。那么互联网医院成品包含哪些功能呢?接下来啊小编就给大家介绍下。1、在线预......
  • 语音社交源码知识语音房间功能的实现
    当今,快节奏的社会为现在的很多年轻人带来了压力,每到深夜或是压力大的时候,很多人都想找人倾诉一下自己心里的悲伤与痛苦,以此来释放一下自己的压力,然而,这是一个独生子女的社会,大部分家庭都只有一个孩子,并且每天工作繁忙又加班到很晚的人根本没有时间或是精力去交朋友,所以这些人没有人......
  • 语音社交源码知识语音房间功能的实现
       当今,快节奏的社会为现在的很多年轻人带来了压力,每到深夜或是压力大的时候,很多人都想找人倾诉一下自己心里的悲伤与痛苦,以此来释放一下自己的压力,然而,这是一个独生子女的社会,大部分家庭都只有一个孩子,并且每天工作繁忙又加班到很晚的人根本没有时间或是精力去交朋友,所以这......
  • jpa2.2.10 JpaSpecificationExecutor Specification 查询时,实体类中存在@OneToMany,怎
    对于这个问题,您可以在查询时通过join()方法指定不查询@OneToMany关联表。举个例子,如果有一个实体A与实体B是一对多关系,那么您可以这样写查询条件:@OneToMany(targetEntity=DcyRightImage.class,cascade={CascadeType.ALL},fetch=FetchType.EAGER)@JoinColumn(nam......