生命周期是组件的重要问题,尤其是在多线程的环境中,比如互联网应用,Socket请求等,而MyBatis也常用于多线程环境中,错误使用会造成严重的多线程并发问题,为了正确编写MyBatis的应用程序,我们需要掌握MyBatis组件的生命周期。
所谓生命周期就是每一个对象应该存活的时间,比如一些对象一次用完后就要关闭,使它们被Java虚拟机(JVM)销毁,以避免继续占用资源,所以我们会根据每一个组件的作用去确定其生命周期。

SqlSessionFactoryBuilder

SqlsessionFactoryBuilder 的作用在于创建SqlSessionFactory,创建成功后,SqlsessionFactoryBuilder 就失去了作用,所以它只能存在于创建SqlSessionFactory 的方法中,而不要让其长期存在。

SqlSessionFactory

SqlSessionFactory 可以被认为是一个数据库连接池,它的作用是创建SqlSession接口对象。因为Mybatis的本质就是Java对数据库的操作,所以SqlSessionFactory的生命周期存在整个MyBatis的应用之中,所以一旦创建了SqlSessionFactory,就要长期保存它,直至不再使用MyBatis应用,所以可以认为SqlSessionFactory 的生命周期就等同于MyBatis的应用周期。

由于SqlSessionFactory 是一个对数据库的连接池,所以它占据着数据库的连接资源,如果创建多个SqlSessionFactory,那么就存在多个数据库连接池,这样不利于对数据库资源的控制,也会导致数据库连接资源被消耗光,出现系统宕机等情况,所以尽量避免发生这样的情况。因此在一般的应用中我们往往希望SqlSessionFactory 作为一个单例,让它在应用中被共享。

SqlSession

如果说SqlSessionFactory 相当于数据库连接池,那么SqlSession就相当于一个数据库连接(Connection 对象),你可以在一个事务里面执行多条SQL,然后通过它的 commit,rollback 等方法,提交或者回滚事务。所以它应该存活在一个业务请求中,处理完整个请求后,应该关闭这条连接,让它归还给SqlSessionFactory,否则数据库资源很快就会被耗费精光,系统就会瘫痪,所以用try...catch...finally语句来保证其正确关闭。

Mapper

Mapper 是一个接口,它由SqlSession 所创建,所以它的最大生命周期至多和SqlSession保持一致,尽管它很好用,但是由于SqlSession 的关闭,它的数据库连接资源也会消失,所以它的生命周期应该小于等于SqlSession 的生命周期。Mapper 代表的是一个请求中的业务处理,所以它应该在一个请求中,一旦处理完了相关的业务,就应该废弃它。

mybatis

Q.E.D.

知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议

悠然心会,妙处难与君说