mybatis的执行流程是什么?

2020-08-05 11:07发布

2条回答
你的回答
2楼 · 2020-08-05 15:05

1. 加载全局配置文件(mybatisConfig.xml),这个配置文件中通常是别名设置,拦截器的设置,

(当ssm整合后,环境配置与mapper映射文件的注册会转移到spring

配置文件中)
2. xml全局配置文件会产生一个构建者类,叫做xmlConfigBuilder,这个类是用来通过xml配置

文件来构建COnfiguration对象实例的,构建的过程就是解析(MBatistConfig

。xml)配置文件  调用parse产生configuration对象
3. 随后产生的就是Mybatis的配置类,(configuration),这个类可以作为项目的全局配置对象,
4. 接下来便是SqlSessionFactory(会话工厂)的构建者类,(SqlSessionFactoryBuilder),

configuration配置对象,就可以调用会话工厂构建者类中的build方法完成对

会话工厂对象的构建
5. 产生sqlsessionFactory(会话工厂),是用来生成会话的接口,有一个实现类(DefaultSqlSessionFactory)

,这个实现类是真正的会话 的工厂类,并且他他是单利的。

会一直存在到服务器关闭
6. 通过调用会话工厂的实现类中(DefaultSqlSessionFactory)的openSession()方法完成SqlSession对象的创建
7. 产生sqlSession,该接口是会话,并且是非线程安全的,每一次对数据库的访问都需要创建一个sqlSession,

当得到结果后SqlSession就会被废弃。所以声明周期短
8.. 当然这当中还有一个Executor执行器接口,这才是内部真正对数据库进行操作的操作者,他才是真正的干事的
9.另外就是StatementHandler该类是Statment处理器,封装了对数据库各种操作方法,使用的时候,

就调用其中的一些方法罢了
10,最后就是结果集处理器(ResultSetHandler),这个处理器的作用就是对结果进行处理并返回的。


我的网名不再改
3楼 · 2020-08-05 16:53

mybatis的执行流程如下代码: 

第一步:通过Resources加载配置好的mybatis.xml配置文件。

Resources是ibatis.io包下面的类,也就是一个io流,用于读写文件,通过getResourceAsStream把xml文件加载进来,把配置文件解析为一个流。

  第二步:然后看第二句话,这句话是关键。我们首先new了一个SqlSessionFactoryBuilder对象,他是SqlSessionFactory的构建者。我们调用了他的build()方法。这个方法长什么样子呢?看一下他的源码:

 

我们发现里面有一个XMLconfigBuilder对象,他是做什么的呢?他是用来解析XML文件的一个构建者,通过他的parse()方法解析mybatis配置文件

解析configuration节点下的子节点: 

 然后我们发现parse()解析完成后,他返回了一个configuration对象,它是用来存放mybatis核心配置文件解析完成后的结果。这个configuration又返回给谁了呢?继续看源码:

又返回了一个build方法,把刚才的返回值configuration作为参数传入这个方法中,并返回了一个DefaultSqlSessionFactory对象,这是SqlSessionFactory的实现类,用来生产defaultSqlSession对象。这样上面的第二句话才算真正的结束了。。。

 这时候会不会有的小伙伴疑惑,为什么既然DefaultSqlSessionFactory是SqlSessionFactory的实现类,为什么我们不直接new一个defaultSqlSessionFactory呢?原因:看源码知道,如果new defaultSqlSessionFactory的话,我们需要一个configuration对象,而这个对象有事经过层层磨难才得到的,所有我们一般采取构建筑模式,直接把xml字节流或字符流给了sqlSessionfactoryBuilder就好了,我们就不用管他是怎么生成的了。

第三步:我们继续往下走,我们最终的目的是获取一个SqlSession对象,现在我们有了一个SqlSessionFactory了,就愉快的生成SqlSession吧。从第三句看,调用了openSession()方法,这又是个啥?看源码:

 

先看到返回值是sqlSession,再看看里面有些什么:

   Transaction,这很明显是一个事务,我们都知道sql执行时都要涉及到事务操作的,提交或回滚什么的。而这个transaction产生需要什么参数呢,看源码知道里面有个environment(xml中配置的),而且是从configuration中获得的(之前做的总算没有白费),通过这些参数transactionFactory就帮我们产生了transaction。

 

executor,这个名字一看就知道是一个执行器,是个接口(一般用SimpleExecutor来执行,实现类),他是mybatis的核心执行器,相当于jdbc中的statement,发送sql语句并执行。

有了这些我们就能得到最后的sqlSession了。

第四步:jdk动态代理生成mapper接口的代理对象

通过DefaultSqlSession的getMapper()来生成

其实是通过mapperRegistry的对象,它里面是一个knownMappers的HashMap,用于存储配置文件中的mappers标签的每个mapper接口

这里通过JDK动态代理,帮mapper接口生成代理实现类

第五步:通过第四步返回的代理对象的getUser方法调用getMapper方法最终执行的方法

代理对象的getUser方法执行其实走的是MapperProxy的invoke方法

这里是判断sql的执行类型 

查询方法 

走executor的query方法 

走CachingExecutor的query方法,先从二级缓存中获取

如二级缓存中没有数据,走delegate(BaseExecutor)的query方法 ,也就是一级缓存localCache,如果一级缓存没有数据,则走queryFromDatabase方法查数据库

从数据库查到数据,放入到一级缓存中 

一级缓存底层是个PerpetualCache类型的HashMap, 

 总结一下:mybatis运行时要先通过resources把核心配置文件也就是mybatis.xml文件加载进来,然后通过xmlConfigBulider来解析,解析完成后把结果放入configuration中,并把它作为参数传入到build()方法中,并返回一个defaultSQLSessionFactory。我们再调用openSession()方法,来获取SqlSession,在构建SqlSession的同时还需要transaction和executor用于后续执行操作。

 


相关问题推荐

  • 回答 2

    Statement的execute(String query)方法用来执行任意的SQL查询,如果查询的结果是一个ResultSet,这个方法就返回true。如果结果不是ResultSet,比如insert或者update查询,它就会返回false。我们可以通过它的getResultSet方法来获取ResultSet,或者通过getUpda...

  • 回答 22

    忙的时候项目期肯定要加班 但是每天加班应该还不至于

  • 回答 108
    已采纳

    虽然Java人才越来越多,但是人才缺口也是很大的,我国对JAVA工程师的需求是所有软件工程师当中需求大的,达到全部需求量的60%-70%,所以Java市场在短时间内不可能饱和。其次,Java市场不断变化,人才需求也会不断增加。马云说过,未来的制造业要的不是石油,...

  • 回答 5
    已采纳

    工信部证书含金量较高。工信部是国务院的下属结构,具有发放资质、证书的资格。其所发放的证书具有较强的权威性,在全国范围内收到认可,含金量通常都比较高。 工信部证书,其含义也就是工信部颁发并承认的某项技能证书,是具有法律效力的,并且是国家认可的...

  • 回答 70
    已采纳

    学Java好不好找工作?看学完Java后能做些什么吧。一、大数据技术Hadoop以及其他大数据处理技术都是用Java或者其他,例如Apache的基于Java 的 HBase和Accumulo以及ElasticSearchas。但是Java在此领域并未占太大空间,但只要Hadoop和ElasticSearchas能够成长壮...

  • 回答 16
    已采纳

    就是java的基础知识啊,比如Java 集合框架;Java 多线程;线程的五种状态;Java 虚拟机;MySQL (InnoDB);Spring 相关;计算机网络;MQ 消息队列诸如此类

  • 回答 12

    #{}和${}这两个语法是为了动态传递参数而存在的,是Mybatis实现动态SQL的基础,总体上他们的作用是一致的(为了动态传参),但是在编译过程、是否自动加单引号、安全性、使用场景等方面有很多不同,下面详细比较两者间的区别:1.#{} 是 占位符 :动态解析 ...

  • 回答 62

    没问题的,专科学历也能学习Java开发的,主要看自己感不感兴趣,只要认真学,市面上的培训机构不少都是零基础课程,能跟得上,或是自己先找些资料学习一下。

  • 回答 4

    1、反射对单例模式的破坏采用反射的方式另辟蹊径实例了该类,导致程序中会存在不止一个实例。解决方案其思想就是采用一个全局变量,来标记是否已经实例化过了,如果已经实例化过了,第 二次实例化的时候,抛出异常2、clone()对单例模式的破坏当需要实现单例的...

  • 回答 5

     优点: 一、实例控制  单例模式会阻止其他对象实例化其自己的单例对象的副本,从而确保所有对象都访问唯一实例。 二、灵活性  因为类控制了实例化过程,所以类可以灵活更改实例化过程。 缺点: 一、开销  虽然数量很少,但如果每次对象请求引用时都要...

  • 回答 4

    这个主要是看你数组的长度是多少, 比如之前写过的一个程序有个数组存的是各个客户端的ip地址:string clientIp[4]={XXX, xxx, xxx, xxx};这个时候如果想把hash值对应到上面四个地址的话,就应该对4取余,这个时候p就应该为4...

  • 回答 6

     哈希表的大小 · 关键字的分布情况 · 记录的查找频率 1.直接寻址法:取关键字或关键字的某个线性函数值为散列地址。即H(key)=key或H(key) = a·key + b,其中a和b为常数(这种散列函数叫做自身函数)。...

  • 回答 6

    哈希表的大小取决于一组质数,原因是在hash函数中,你要用这些质数来做模运算(%)。而分析发现,如果不是用质数来做模运算的话,很多生活中的数据分布,会集中在某些点上。所以这里最后采用了质数做模的除数。 因为用质数做了模的除数,自然存储空间的大小也用质数了...

  • 回答 2

    是啊,哈希函数的设计至关重要,好的哈希函数会尽可能地保证计算简单和散列地址分布均匀,但是,我们需要清楚的是,数组是一块连续的固定长度的内存空间

  • 回答 3

     解码查表优化算法,seo优化

  • 回答 5

    1.对对象元素中的关键字(对象中的特有数据),进行哈希算法的运算,并得出一个具体的算法值,这个值 称为哈希值。2.哈希值就是这个元素的位置。3.如果哈希值出现冲突,再次判断这个关键字对应的对象是否相同。如果对象相同,就不存储,因为元素重复。如果对象不同,就...

没有解决我的问题,去提问