当前位置: 首页 > news >正文

MyBatis源码解读4(2.3、MyBatis运行流程)

2.3、MyBatis运行流程

2.3.1、配置解析

 InputStream inputStream = Resources.getResourceAsStream("mybatis-config.xml");

通过IO方式打开输入流,获取mybatis-config.xml以及xxxMapper.xml,因为在mybatis-config.xml中已经配置好了mapper文件的路径,所以在以一次的IO中把两者的信息都读到了。

那么在读取完配置文件以后,MyBatis该如何处理所读取到的这些配置信息呢?在读取完mybatis-config.xml文件后,所有的配置都将会被封装成Configuration对象。那么此时问题又来了,如何将xml对象转为java对象呢?java提供了xml解析的一整套完整的结局方案。

xml解析在java中有3种方式:

  1. dom
  2. sax
  3. xpath

而MyBatis采用的是第三种方式,MyBatis在解析的时候使用的对象是XPathParser对象,利用这个对象来读取、解析xml,再进一步封装成XNode对象,这个XNode对象对应的是xml文件中的的一个个标签,通过这两个对象的组合使用来解析整个xml文件。

在读取配置文件的这一行代码肯定会包含解析xml配置文件的步骤,所以我们跟进来看看这个build方法。

SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);

image-20230624122704490

接着点这个build方法。

image-20230624122727105

可以看到他是有解析这个xml的方法。我们点进去

image-20230628215228615

再点进去,我们会看到一个parseConfiguration方法,这个方法用于解析一个个的标签。

image-20230628215333713

随便抽出一行来解析可以看到,这个方法是用于解析mappers标签的方法。

 mapperElement(root.evalNode("mappers"));

image-20230628215534502

我们继续溯源,点mapperElement方法来剥丝抽茧。

image-20230628220148101

我们在写mappers标签的时候,如果写了packages属性(写dao接口所在的包的路径)的话,就不用再单独写一个个的dao类的路径,因为他会去扫描这个路径所在的所有的类。

 if ("package".equals(child.getName())) {String mapperPackage = child.getStringAttribute("name");configuration.addMappers(mapperPackage);}

如果我们不写packeags这个属性的话,就会走到else分支里面。

	  String resource = child.getStringAttribute("resource");String url = child.getStringAttribute("url");String mapperClass = child.getStringAttribute("class");if (resource != null && url == null && mapperClass == null) {ErrorContext.instance().resource(resource);try (InputStream inputStream = Resources.getResourceAsStream(resource)) {XMLMapperBuilder mapperParser = new XMLMapperBuilder(inputStream, configuration, resource,configuration.getSqlFragments());mapperParser.parse();}} else if (resource == null && url != null && mapperClass == null) {ErrorContext.instance().resource(url);try (InputStream inputStream = Resources.getUrlAsStream(url)) {XMLMapperBuilder mapperParser = new XMLMapperBuilder(inputStream, configuration, url,configuration.getSqlFragments());mapperParser.parse();}} else if (resource == null && url == null && mapperClass != null) {Class<?> mapperInterface = Resources.classForName(mapperClass);configuration.addMapper(mapperInterface);} else {throw new BuilderException("A mapper element may only specify a url, resource or class, but not more than one.");}

前三行其实是在获取我们写的属性。

	 String resource = child.getStringAttribute("resource");String url = child.getStringAttribute("url");String mapperClass = child.getStringAttribute("class");

如果我们这样写标签的话

 <mapper resource="cn/linstudy/mapper/DeptMapper.xml"></mapper>

那么resource的值就是cn/linstudy/mapper/DeptMapper.xml,而url、mapperClass的值将会全部为空。然后就到了第一个if判断语句。

if (resource != null && url == null && mapperClass == null) {ErrorContext.instance().resource(resource);try (InputStream inputStream = Resources.getResourceAsStream(resource)) {XMLMapperBuilder mapperParser = new XMLMapperBuilder(inputStream, configuration, resource,configuration.getSqlFragments());mapperParser.parse();}
}

我们看到这个if条件语句可以得知,mapper标签里面的属性,只有resource是不可以为空,其他均可以为空。如果resource不为空的话就可以获取到这个属性对应的值。

然后我们就可以开始看parse这个方法,用于解析。

image-20230628224613423

然而负责解析maapper的是第三行。

 configurationElement(parser.evalNode("/mapper"));

所以我们接着顺藤摸瓜。

image-20230628224748278

我们可以看到这个方法是真正开始读取我们的xxxMapper.xml文件,也就是读取我们写的一些sql语句了。

其中这个方法最重要的是buildStatementFromContext(context.evalNodes("select|insert|update|delete"));这一句,我们点进去buildStatementFromContext方法。

image-20230628225332250

其中statementParser的parseStatementNode是将传过来的参数解析成mapperstatememt,我们来看看parseStatementNode方法。

image-20230628225736597

image-20230628225857751

我们看到这些红框标出来的有没有感到特别的熟悉呢?

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

​ 没错这些就是在mapper.xml文件中写的每一个标签里面的属性。

image-20230628230037064

​ 我们再返回到parseStatementNode这个方法,我们可以发现前面的步骤都是在为了120行的addMappedStatement方法做数据准备。

image-20230628230235108

​ 所以解析了那么多标签里面的数据,其实最终的·目的都是为了addMappedStatement来创建一个mapperstatement来准备数据的。我们进入这个方法。

image-20230628230438484

​ 我们可以发现一个很经典的设计模式——构造者设计模式,构造者设计模式其实本质上是为了创建对象的,但凡使用了创建者模式创建对象的话,他创建对象的方法都是叫做build。

image-20230628230524893

​ 这里就创建了一个MappedStatement对象,使用的也是创建者模式。

image-20230629222607896

​ 在创建完mapperstatement对象后就存入到configuration对象里面去了。

    MappedStatement statement = statementBuilder.build();configuration.addMappedStatement(statement);

2.3.1、SqlSessionFactory

​ 了解完配置的解析以后,我们来看看SqlSessionFactory的创建。我们找到我们写过的测试类,有一个build方法可以点进去。

image-20230629224217866

​ 再点这个build方法。

image-20230629224242855

​ 好家伙,还有一个build方法,我们可以看到build方法创建了一个默认的DefaultSqlSessionFactory对象。

image-20230629224319595

2.3.2、openSession

​ 我们看完SqlSessionFactory创建以后,来看看sqlSessionFactory的openSession方法。

image-20230629224739526

​ 然后我们发现这是一个接口方法。

image-20230629224836912

​ 我们找第一个实现,因为SqlSessionFactory是由DefaultSqlSessionFactory创建的,所以我们找DefaultSqlSessionFactory实现的方法即可。

image-20230629224919310

​ 点进去后是一个openSessionFromDataSource方法,我们可以发现一个特别有意思的现象,我们来看传入的参数是一个ExecutorType,也就是说虽然说是openSession,但是实际上真正干活的是Executor

image-20230629225052336

​ 然后我们继续点进去这个方法,接着跟,这个就是创建sqlSession的核心。

image-20230702140622456

http://www.xdnf.cn/news/396685.html

相关文章:

  • 当虚拟吞噬现实——《GTA6》结合技术
  • 每日算法-250511
  • 广东省省考备考(第八天5.11)—言语:逻辑填空(每日一练)
  • AMD FPGA书籍推荐-初学者、一线工程师适用
  • 共享内存与信号量结合
  • xilinx QDMA开发调试记录
  • 《算法导论(第4版)》阅读笔记:p18-p31
  • NB-IoT嵌入式产品开发有哪些坑?
  • 基于 TSBS 标准数据集下 TimescaleDB、InfluxDB 与 TDengine 性能对比测试报告
  • 【八股消消乐】项目中如何排查内存持续上升问题
  • 英伟达推理模型论文速读:OpenCodeReasoning-Nemotron-32B
  • 信息学奥赛一本通 1488:新的开始
  • C++之红黑树
  • TypeScript 中的泛型工具详解
  • HVV面试题汇总合集
  • 万字了解什么是微前端???
  • 滑动窗口:穿越数据的时光机
  • YOLOv11与Roboflow数据集使用全攻略
  • Linux : 31个普通信号含义
  • LlamaIndex 第七篇 结构化数据提取
  • Java常用类-String三剑客
  • 不换设备秒通信,PROFINET转Ethercat网关混合生产线集成配置详解
  • iVX:图形化编程与组件化的强强联合
  • CSS 盒子模型与元素定位
  • 汽车诊断简介
  • 【Linux高级全栈开发】2.1高性能网络-网络编程——2.1.1 网络IO与IO多路复用——select/poll/epoll
  • 1、虚拟人物角色聊天 AI Agent 设计方案
  • FME处理未知或动态结构教程
  • FPGA生成随机数的方法
  • 2505d,d的一些疑问