mybatis 调用 oracle函数_MyBatis之启动分析(一)
作者丨ytao来源丨ytao(ytao-blog)前言MyBatis 作为目前最常用的持久层框架之一,分析其源码,对我们的使用过程中可更好的运用它。本系列基于 mybatis-3.4.6进行分析。MyBatis 的初始化工作就是解析主配置文件,映射配置文件以及注解信息。然后保存在 org.apache.ibatis.session.Configuration,供后期执行数据请求的相关调...
·
作者丨ytao 来源丨ytao(ytao-blog)
前言
MyBatis 作为目前最常用的持久层框架之一,分析其源码,对我们的使用过程中可更好的运用它。本系列基于mybatis-3.4.6进行分析。MyBatis 的初始化工作就是解析主配置文件,映射配置文件以及注解信息。然后保存在 org.apache.ibatis.session.Configuration,供后期执行数据请求的相关调用。Configuration 里有大量配置信息,在后面每涉及到一个相关配置,会进行详细的分析。
启动
publicstaticvoid main(String[] args) throwsIOException{// 获取配置文件Reader reader = Resources.getResourceAsReader("mybatis-config.xml");// 通过 SqlSessionFactoryBuilder 构建 sqlSession 工厂SqlSessionFactory sqlSessionFactory = newSqlSessionFactoryBuilder().build(reader);// 获取 sqlSession 实例SqlSession sqlSession = sqlSessionFactory.openSession(); reader.close(); sqlSession.close();}
分析
SqlSessionFactoryBuilder 类
SqlSessionFactoryBuilder 的build()是Mybatis启动的初始化入口,使用builder模式加载配置文件。通过查看该类,使用方法重载,有以下9个方法:
方法重载最终实现处理的方法源码如下:
publicSqlSessionFactory build(Reader reader, String environment, Properties properties) {try{// 实例化 XMLConfigBuilder,用于读取配置文件信息XMLConfigBuilder parser = newXMLConfigBuilder(reader, environment, properties);// 解析配置信息,保存到 Configurationreturn build(parser.parse());} catch(Exception e) {throwExceptionFactory.wrapException("Error building SqlSession.", e);} finally{ErrorContext.instance().reset();try{ reader.close();} catch(IOException e) {// Intentionally ignore. Prefer previous error.}}}
- environment 是指定加载环境,默认值为 null。
- properties 是属性配置文件,默认值为 null。同时读取配置文件既可字符流读取,也支持字节流读取。
publicSqlSessionFactory build(InputStream inputStream, String environment, Properties properties) {try{XMLConfigBuilder parser = newXMLConfigBuilder(inputStream, environment, properties);return build(parser.parse());} catch(Exception e) {throwExceptionFactory.wrapException("Error building SqlSession.", e);} finally{ErrorContext.instance().reset();try{ inputStream.close();} catch(IOException e) {// Intentionally ignore. Prefer previous error.}}}
实例化 XMLConfigBuilder 类
通过 SqlSessionFactoryBuilder 中XMLConfigBuilderparser=newXMLConfigBuilder(reader,environment,properties), 分析 XMLConfigBuilder实例化过程。该类中有四个变量:
privateboolean parsed;privatefinalXPathParser parser;privateString environment;privatefinalReflectorFactory localReflectorFactory = newDefaultReflectorFactory();
- parsed 是否解析,一次解析即可。用于标志配置文件只解析一次,
true为已解析过。 - parser 解析配置的解析器
- environment 加载环境,即
SqlSessionFactoryBuilder中的environment - localReflectorFactory 用于创建和缓存
Reflector对象,一个类对应一个Reflector。因为参数处理、结果映射等操作时,会涉及大量的反射操作。DefaultReflectorFactory实现类比较简单,这里不再进行讲解。
publicXMLConfigBuilder(Reader reader, String environment, Properties props) {this(newXPathParser(reader, true, props, newXMLMapperEntityResolver()), environment, props);}
实例化 XPathParser 对象
首先实例化 XPathParser 对象,里面定义了5个变量:
privatefinalDocument document;privateboolean validation;privateEntityResolver entityResolver;privateProperties variables;privateXPath xpath;
- document 保存document对象
- validation xml解析时是否验证文档
- entityResolver 加载dtd文件
- variables 配置文件定义的值
- xpath Xpath对象,用于对XML文件节点的操作
XPathParser 对象构造函数有:
函数里面都处理了两件事:
publicXPathParser(Reader reader, boolean validation, Properties variables, EntityResolver entityResolver) { commonConstructor(validation, variables, entityResolver);this.document = createDocument(newInputSource(reader));}
- 初始化赋值,和创建
XPath对象,用于对XML文件节点的操作。
privatevoid commonConstructor(boolean validation, Properties variables, EntityResolver entityResolver) {this.validation = validation;this.entityResolver = entityResolver;this.variables = variables;// 创建Xpath对象,用于对XML文件节点的操作XPathFactory factory = XPathFactory.newInstance();this.xpath = factory.newXPath();}
- 创建
Document对象并赋值到document变量, 这里属于Document创建的操作,不再详细讲述,不懂可以点击这里查看API (https://docs.oracle.com/javase/8/docs/api/org/w3c/dom/Document.html?is-external=true)
privateDocument createDocument(InputSource inputSource) {// important: this must only be called AFTER common constructortry{// 实例化 DocumentBuilderFactory 对象,用于创建 DocumentBuilder 对象DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();// 是否校验文档 factory.setValidating(validation);// 设置 DocumentBuilderFactory 的配置 factory.setNamespaceAware(false); factory.setIgnoringComments(true); factory.setIgnoringElementContentWhitespace(false); factory.setCoalescing(false); factory.setExpandEntityReferences(true);// 创建 DocumentBuilderDocumentBuilder builder = factory.newDocumentBuilder(); builder.setEntityResolver(entityResolver); builder.setErrorHandler(newErrorHandler() {@Overridepublicvoid error(SAXParseException exception) throwsSAXException{throw exception;}@Overridepublicvoid fatalError(SAXParseException exception) throwsSAXException{throw exception;}@Overridepublicvoid warning(SAXParseException exception) throwsSAXException{}});// 加载文件return builder.parse(inputSource);} catch(Exception e) {thrownewBuilderException("Error creating document instance. Cause: "+ e, e);}}
XMLConfigBuilder构造函数赋值
privateXMLConfigBuilder(XPathParser parser, String environment, Properties props) {super(newConfiguration());ErrorContext.instance().resource("SQL Mapper Configuration");this.configuration.setVariables(props);this.parsed = false;this.environment = environment;this.parser = parser;}
- 初始化父类
BaseBuilder的值。 - 将外部值赋值给对象。
- 将实例化的
XPathParser赋值给parser。
XMLConfigBuilder对象。
解析 XMLConfigBuilder 对象
通过XMLConfigBuilder.parse() 解析配置信息,保存至 Configuration。解析详解在后面文章中进行分析。
publicConfiguration parse() {// 是否解析过配置文件if(parsed) {thrownewBuilderException("Each XMLConfigBuilder can only be used once.");}// 标志解析过,定义为 true parsed = true;// 解析 configuration 节点中的信息 parseConfiguration(parser.evalNode("/configuration"));return configuration;}
创建 SqlSessionFactory
DefaultSqlSessionFactory实现了 SqlSessionFactory接口。通过上面解析得到的 Configuration,调用 SqlSessionFactoryBuilder.build(Configurationconfig)创建一个 DefaultSqlSessionFactory。
publicSqlSessionFactory build(Configuration config) {returnnewDefaultSqlSessionFactory(config);}实例化 DefaultSqlSessionFactory的过程,就是将 Configuration传递给 DefaultSqlSessionFactory成员变量 configuration。
publicDefaultSqlSessionFactory(Configuration configuration) {this.configuration = configuration;}
创建 SqlSession
通过调用SqlSessionFactory.openSession()创建 SqlSession。
publicinterfaceSqlSessionFactory{// 默认创建SqlSession openSession();SqlSession openSession(boolean autoCommit);SqlSession openSession(Connection connection);SqlSession openSession(TransactionIsolationLevel level);SqlSession openSession(ExecutorType execType);SqlSession openSession(ExecutorType execType, boolean autoCommit);SqlSession openSession(ExecutorType execType, TransactionIsolationLevel level);SqlSession openSession(ExecutorType execType, Connection connection);Configuration getConfiguration();}
- autoCommit 是否自动提交事务,
- level 事务隔离级别(共5个级别), 可查看相关源码
- connection 连接
- execType 执行器的类型:
SIMPLE(不做特殊处理),REUSE(复用预处理语句),BATCH(会批量执行)
DefaultSqlSessionFactory实现了 SqlSessionFactory接口,所以进入到 DefaultSqlSessionFactory查看 openSession()。
publicSqlSession openSession() {return openSessionFromDataSource(configuration.getDefaultExecutorType(), null, false);}openSession()方法最终实现代码如下:
privateSqlSession openSessionFromDataSource(ExecutorType execType, TransactionIsolationLevel level, boolean autoCommit) {Transaction tx = null;try{// 获取configuration中的加载环境finalEnvironment environment = configuration.getEnvironment();// 获取事务工厂finalTransactionFactory transactionFactory = getTransactionFactoryFromEnvironment(environment);// 创建一个事务 tx = transactionFactory.newTransaction(environment.getDataSource(), level, autoCommit);// 生成一个处理器,事务保存在处理器 BaseExecutor 中finalExecutor executor = configuration.newExecutor(tx, execType);// 实例化一个 DefaultSqlSession,DefaultSqlSession实现了SqlSession接口returnnewDefaultSqlSession(configuration, executor, autoCommit);} catch(Exception e) {// 异常情况下关闭事务 closeTransaction(tx); // may have fetched a connection so lets call close()throwExceptionFactory.wrapException("Error opening session. Cause: "+ e, e);} finally{// 充值错误实例上下文ErrorContext.instance().reset();}}生成处理器 Configuration.newExecutor(Transactiontransaction,ExecutorTypeexecutorType):
publicExecutor newExecutor(Transaction transaction, ExecutorType executorType) {// 默认为 ExecutorType.SIMPLE executorType = executorType == null? defaultExecutorType : executorType; executorType = executorType == null? ExecutorType.SIMPLE : executorType;Executor executor;if(ExecutorType.BATCH == executorType) { executor = newBatchExecutor(this, transaction);} elseif(ExecutorType.REUSE == executorType) { executor = newReuseExecutor(this, transaction);} else{ executor = newSimpleExecutor(this, transaction);}if(cacheEnabled) { executor = newCachingExecutor(executor);} executor = (Executor) interceptorChain.pluginAll(executor);return executor;}以 ExecutorType.SIMPLE为例, BatchExecutor, ReuseExecutor同理:
至此,mybatis的启动流程大致简单的介绍到这里,对mybatis的启动初始化有个大致了解。接下将会针对单独模块进行详细分析。

近期精彩内容推荐:
直播界要哭了!罗永浩进军电商直播
人家裁员我加薪,他凭什么身价1200亿?
什么?你还在使用fastjson,性能太差了
2020年抖音用户画像报告

在看点这里
好文分享给更多人↓↓
魔乐社区(Modelers.cn) 是一个中立、公益的人工智能社区,提供人工智能工具、模型、数据的托管、展示与应用协同服务,为人工智能开发及爱好者搭建开放的学习交流平台。社区通过理事会方式运作,由全产业链共同建设、共同运营、共同享有,推动国产AI生态繁荣发展。
更多推荐



所有评论(0)