springboot集成mybatis源码分析-mybatis的mapper执行查询时的流程(三)
例:
package com.example.demo.service;
import com.example.demo.dao.UserDao;
import com.example.demo.domain.User;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import java.util.List;
@Service
@Slf4j
public class UserService {
@Autowired
private UserDao userDao;
public List getUser(){
List userList = userDao.getUser();
log.info("查询出来的用户信息,{}",userList.toString());
return userList;
}
}
当userService中的getUser方法执行的时候,userDao.getUser()是怎么走的?
1、在springboot项目启动时,加载mybatis相关配置,同事会在MapperRegister中保存mapper的代理类,在创建UserService bean的时候,需要注入userDao类,但userDao类是一个Interface类型,所以在注入的时候其实是注入的一个mapper代理类,也就是MapperProxy类
2、当执行userDao.getUser()时,会走MapperProxy中的invoke方法,最终是通过mapperMethod.execute(sqlSession. args)
data:image/s3,"s3://crabby-images/557b0/557b0d128142c7f39d3a4814f47dd6c3c30bb036" alt="springboot集成mybatis源码分析-mybatis的mapper执行查询时的流程(三)_第1张图片"
3、进入MapperMethod的execute方法,会根据select、insert、update 来走不同的方法
data:image/s3,"s3://crabby-images/53908/539087b883a18f5ca2048a9244551f0c5392d084" alt="springboot集成mybatis源码分析-mybatis的mapper执行查询时的流程(三)_第2张图片"
4、本次测试时走的是SELECT 的case,在该case中会判断是否有返回值(返回值得处理器),因本次查询返回值是一个list,所以会走executeForMany()
data:image/s3,"s3://crabby-images/c328b/c328b8b98b130061f3ea94ebd73abcca592787c6" alt="springboot集成mybatis源码分析-mybatis的mapper执行查询时的流程(三)_第3张图片"
5、进入到executeForMany中,参数是SQLSessionTemplate和null,进行处理后,进入到sqlsession.selectList(*, *)中
data:image/s3,"s3://crabby-images/ec160/ec160c7a4f16eac302f308d1aedc7578e5f2c715" alt="springboot集成mybatis源码分析-mybatis的mapper执行查询时的流程(三)_第4张图片"
6、该selectList方法调用的是SQLSessionTemplate类的selectList
data:image/s3,"s3://crabby-images/83b1d/83b1da720fc11ba5a3ff102db9fe9853d6288e0b" alt=""
7、进入到sqlSessionProxy(该代理类其实就是DefaultSqlSession)的selectList方法中,参数statement就是mapper方法,paramter为null,
在selectList方法中,会从Configuration对象中获取statement对应的对象,然后通过执行器executor的query来执行
data:image/s3,"s3://crabby-images/bd9d4/bd9d47d42d03905c8a1d5688f820312e21af59d3" alt="springboot集成mybatis源码分析-mybatis的mapper执行查询时的流程(三)_第5张图片"
8、进入到executor的query方法中,当前的executor默认是CachingExecutor
data:image/s3,"s3://crabby-images/94594/94594c854688811b1f83f67c6497a59295ec3ac5" alt="springboot集成mybatis源码分析-mybatis的mapper执行查询时的流程(三)_第6张图片"
9、进入到CachingExecutor的query方法中,在当前方法中会先获取相应的BindSql,然后会创建cache
data:image/s3,"s3://crabby-images/14eee/14eeeb2fdf18a203f14f3395bf659bd6381adf8e" alt=""
data:image/s3,"s3://crabby-images/983a3/983a36c580cb9d5bf28125ba9af323a4a16a081d" alt="springboot集成mybatis源码分析-mybatis的mapper执行查询时的流程(三)_第7张图片"
10、进入到createCacheKey方法中,该方法其实是BaseExecutor中的方法
data:image/s3,"s3://crabby-images/e7f01/e7f0178da67b156e34d73f8fac4473e6974f64fb" alt="springboot集成mybatis源码分析-mybatis的mapper执行查询时的流程(三)_第8张图片"
11、创建完成之后,会调用query方法,在方法中,会先查询cache,没有再调用delegate的query方法
data:image/s3,"s3://crabby-images/810b1/810b17c98dea8b4f60f0c09c0c41b357b5cd0ead" alt="springboot集成mybatis源码分析-mybatis的mapper执行查询时的流程(三)_第9张图片"
12、delegate是BaseExecutor(执行器),在当前方法中也会先从缓存中查询,查询不到在从库中查询
data:image/s3,"s3://crabby-images/37a1e/37a1e444ac513c619b79aa14bb184020ae796e2a" alt="springboot集成mybatis源码分析-mybatis的mapper执行查询时的流程(三)_第10张图片"
13、进入queryFormDatabase方法中,在方法中会将之前创建的cacheKey保存到localCache中,然后在执行doQuery方法
data:image/s3,"s3://crabby-images/cd83e/cd83e2a9447d8444863e9ac1b80d140b930f1678" alt="springboot集成mybatis源码分析-mybatis的mapper执行查询时的流程(三)_第11张图片"
14、进入到doQuery方法中,也就是默认的simpleExecutor中
data:image/s3,"s3://crabby-images/f9ec8/f9ec8ea9770eefa93e34caee61a4c3469f11375c" alt="springboot集成mybatis源码分析-mybatis的mapper执行查询时的流程(三)_第12张图片"
15、进入到configuration.newStatementHandler中(默认statement为preparedStatement)
data:image/s3,"s3://crabby-images/aba20/aba209ba80abbb3796ca46ce3fd7187ef83c8782" alt="springboot集成mybatis源码分析-mybatis的mapper执行查询时的流程(三)_第13张图片"
16、返回preparestatementHandler处理器,然后调用prepareStatement方法
data:image/s3,"s3://crabby-images/1a98d/1a98dab9da8ae19a2e40314cb5ef24dd26f094f4" alt="springboot集成mybatis源码分析-mybatis的mapper执行查询时的流程(三)_第14张图片"
data:image/s3,"s3://crabby-images/7af63/7af63691477820ed98ee6b9a23e0ee3d7551feb4" alt="springboot集成mybatis源码分析-mybatis的mapper执行查询时的流程(三)_第15张图片"
18、在方法中会通过prepareStatement的execute来查询数据库,完毕后,会在handeler中处理返回数据
data:image/s3,"s3://crabby-images/78e86/78e863dd52a31697cd3bfa80751fff40d99b466e" alt=""
19、处理返回数据是在DefaultResultSetHandler类中
data:image/s3,"s3://crabby-images/825be/825be9f2a5d6ae0fd1d53778fed8106d42a016a1" alt="springboot集成mybatis源码分析-mybatis的mapper执行查询时的流程(三)_第16张图片"
20、处理完成后悔返回一个list