Mybatis有几个基础:
SqlSession,代表和数据库的一次会话,提供了操作数据库的方法
MapperedStatement,代表要发展数据执行的命令,也就是我们sql语句的丑抽象
Executor。和数据库交互的执行器,接受MapperedStatement
映射接口,在接口中会有要执行的SQl用一个方法表示,具体SQL写在映射文件中。
映射文件,编写SQL的文件
在代码运行时,有可能会在一个数据库会话中执行多次相同的SQL,这种反复的查询会带来开销。
每一个SqlSession都持有自己的缓存,就是LocalCache。一种是Session级别的,在Mybatis中执行的所有语句都会共享这一个缓存。一种是Statement级别的,缓存只对当前statement语句。
当发起查询时,Mybatis会根据statementId、params、rowBounds生成一个key,去Cache查,如果没有的话会去数据库查,并且缓存到本地。
一级缓存的不足 使用一级缓存的时候,因为缓存不能跨会话共享,不同的会话之间对于相同的数据可能有不一样的缓存。在有多个会话或者分布式环境下,会存在脏数据的问题。如果要解决这个问题,就要用到二级缓存。MyBatis?一级缓存无法关闭,但是有两种级别可选:
session级别的缓存,在同一个sqlSession内,对同样的查询将不再查询数据库,直接从缓存中取。
statement级别的缓存,为了避免上述问题,可以将一级缓存的级别设为?statement?级别的,这样每次查询结束都会清掉一级缓存。
二级缓存 二级缓存是用来解决一级缓存不能跨会话共享的问题的,在XML文件配置后可以被多个SqlSession共享,生命周期和应用同步。
如果MyBatis使用了二级缓存,并且Mapper和select语句也配置使用了二级缓存,那么在执行select查询的时候,MyBatis会先从二级缓存中取输入,其次才是一级缓存。
即MyBatis查询数据的顺序是:二级缓存?—>?一级缓存?—>?数据库。
一、二级缓存的使用注意点 MyBatis默认的session级别一级缓存,由于Spring?Boat?中默认使用了hikariCP,所以基本没用,需要开启事务才有用。但一级缓存作用域仅限同一sqlSession内,无法感知到其他sqlSession的增删改,所以极易产生脏数据。
二级缓存可通过cache-ref让多个mapper.xml共享同一namespace,从而实现缓存共享,但多表联查时配置略微繁琐。
生产环境建议将一级缓存设置为statment级别(即关闭一级缓存),如果有必要,可以开启二级缓存。
在分布式环境中也是不建议开启二级缓存的,因为缓存是保存到本地的,这样也会导致产生脏数据。 ?
|