mybatis_helloworld(2)_源码
摘录自:http://blog.csdn.net/y172158950/article/details/16982123
在helloworld(1)中,分析了insert一条数据的流程,现在分析下源码:
- public static void main(String args[]) throws IOException {
- String resource = "com/test/configuration.xml";//获得xml(Mybatis)数据库连接的连接
- Reader reader = Resources.getResourceAsReader(resource);//读取里面的文件
- SqlSessionFactory sessionFactory = new SqlSessionFactoryBuilder()
- .build(reader);//创建session工厂
- SqlSession session = sessionFactory.openSession();//打开session
- new Test1().insert(session);
- }
1. XML文件资源的获取
a)包路径:org.apache.ibatis.io
b)Resources类:定义了一系列的静态方法,用于获取资源(ClassLoader,URL,InputStream,Read,File,Properties,Charset)
c)我们用到的getResourceAsReader,getResourceAsStream方法
- <p> public static Reader getResourceAsReader(String resource)
- throws IOException
- {
- Reader reader;</p><p>· //利用InputStream构造Reader
- if(charset == null)
- reader = new InputStreamReader(getResourceAsStream(resource));
- else
- reader = new InputStreamReader(getResourceAsStream(resource), charset);
- return reader;
- }</p><p> </p><p> public static InputStream getResourceAsStream(String resource)
- throws IOException
- {</p><p> //调用重载方法
- return getResourceAsStream(null, resource);
- }</p><p> </p><p> public static InputStream getResourceAsStream(ClassLoader loader, String resource)
- throws IOException
- {</p><p> //调用ClassLoaderWrapper类的方法,利用ClassLoader获取资源
- InputStream in = classLoaderWrapper.getResourceAsStream(resource, loader);
- if(in == null)
- throw new IOException((new StringBuilder()).append("Could not find resource ").append(resource).toString());
- else
- return in;
- }</p>
d)ClassLoaderWrapper类:封装了一个ClassLoader数组,通过ClassLoader获取资源
e)我们用到的getResourceAsStream,getClassLoaders方法
- public InputStream getResourceAsStream(String resource, ClassLoader classLoader)
- {
- return getResourceAsStream(resource, getClassLoaders(classLoader));
- }
- ClassLoader[] getClassLoaders(ClassLoader classLoader)
- {
- return (new ClassLoader[] { //构建新的ClassLoader列表,包含用户创建及默认
- classLoader, defaultClassLoader, Thread.currentThread().getContextClassLoader(), getClass().getClassLoader(), systemClassLoader});
- }
- InputStream getResourceAsStream(String resource, ClassLoader classLoader[])
- {
- ClassLoader arr$[] = classLoader;
- int len$ = arr$.length;
- for(int i$ = 0; i$ < len$; i$++) //遍历ClassLoader数组,当某一个Loader成功构建资源则返回
- {
- ClassLoader cl = arr$[i$];
- if(null == cl)
- continue;
- InputStream returnValue = cl.getResourceAsStream(resource);
- if(null == returnValue)
- returnValue = cl.getResourceAsStream((new StringBuilder()).append("/").append(resource).toString());
- if(null != returnValue)
- return returnValue;
- }
- return null;
- }
2. 构建SqlSessionFactory,初始化资源
a) SqlSessionFactory接口介绍
- public interface SqlSessionFactory
- {
- //定义了一系列获取Session的方法,获取Configuration的方法
- public abstract SqlSession openSession();
- public abstract SqlSession openSession(boolean flag);
- public abstract SqlSession openSession(Connection connection);
- public abstract SqlSession openSession(TransactionIsolationLevel transactionisolationlevel);
- public abstract SqlSession openSession(ExecutorType executortype);
- public abstract SqlSession openSession(ExecutorType executortype, boolean flag);
- public abstract SqlSession openSession(ExecutorType executortype, TransactionIsolationLevel transactionisolationlevel);
- public abstract SqlSession openSession(ExecutorType executortype, Connection connection);
- public abstract Configuration getConfiguration();
- }
b) SqlSessionFactory接口构建
i. SqlSessionFactoryBuilder类:构建SqlSessionFactory
ii. 我们使用的方法:build
- public SqlSessionFactory build(Reader reader)
- {
- //调用重载方法
- return build(reader, null, null);
- }
- public SqlSessionFactory build(Reader reader, String environment, Properties properties)
- {
- try
- {
- SqlSessionFactory sqlsessionfactory;
- try
- {
- //根据reader封装XMLConfig相关信息
- XMLConfigBuilder parser = new XMLConfigBuilder(reader, environment, properties);
- //调用重载方法
- sqlsessionfactory = build(parser.parse());
- }
- catch(Exception e)
- {
- throw ExceptionFactory.wrapException("Error building SqlSession.", e);
- }
- return sqlsessionfactory;
- }
- finally
- {
- ErrorContext.instance().reset();
- try
- {
- reader.close();
- }
- catch(IOException e) { }
- }
- }
- public SqlSessionFactory build(Configuration config)
- {
- //依据配置信息构建默认的SqlSessionFactory实现
- return new DefaultSqlSessionFactory(config);
- }
c) XMLConfigBuilder类:解析XML配置文件
i. 用到的主要方法:parse,parseConfiguration(就是按XML结构顺序解析,分别获取配置[SAX解析]);将数据封装到父类的configuration对象中
- public Configuration parse()
- {
- if(parsed)
- {
- throw new BuilderException("Each MapperConfigParser can only be used once.");
- } else
- {
- parsed = true;
- parseConfiguration(parser.evalNode("/configuration")); //XML文件的根目录
- return configuration;
- }
- }
- private void parseConfiguration(XNode root)
- {
- try
- {
- typeAliasesElement(root.evalNode("typeAliases")); //解析typeAliases
- pluginElement(root.evalNode("plugins"));
- objectFactoryElement(root.evalNode("objectFactory"));
- objectWrapperFactoryElement(root.evalNode("objectWrapperFactory"));
- propertiesElement(root.evalNode("properties"));
- settingsElement(root.evalNode("settings"));
- environmentsElement(root.evalNode("environments")); //数据库配置
- typeHandlerElement(root.evalNode("typeHandlers"));
- mapperElement(root.evalNode("mappers")); //解析sql语句配置文件
- }
- catch(Exception e)
- {
- throw new BuilderException((new StringBuilder()).append("Error parsing SQL Mapper Configuration. Cause: ").append(e).toString(), e);
- }
- }
d)Configuration类:基本就是封装一系列的数据;看一下成员变量
- protected Environment environment;
- protected boolean lazyLoadingEnabled;
- protected boolean aggressiveLazyLoading;
- protected boolean multipleResultSetsEnabled;
- protected boolean useGeneratedKeys;
- protected boolean useColumnLabel;
- protected boolean cacheEnabled;
- protected Integer defaultStatementTimeout;
- protected ExecutorType defaultExecutorType;
- protected AutoMappingBehavior autoMappingBehavior;
- protected Properties variables;
- protected ObjectFactory objectFactory;
- protected ObjectWrapperFactory objectWrapperFactory;
- protected MapperRegistry mapperRegistry;
- protected final InterceptorChain interceptorChain;
- protected final TypeHandlerRegistry typeHandlerRegistry;
- protected final TypeAliasRegistry typeAliasRegistry;
- protected final Map mappedStatements;
- protected final Map caches;
- protected final Map resultMaps;
- protected final Map parameterMaps;
- protected final Map keyGenerators;
- protected final Set loadedResources;
- protected final Map sqlFragments;
- protected final Collection incompleteStatements;
- protected final Collection incompleteCacheRefs;
- protected final Map cacheRefMap;
e)DefaultSqlSessionFactory类:SqlSessionFactory接口实现类,并持有一个Configuration对象;我们实现获取SqlSeesion对象应用的此类的方法。
3. 回过头来,再看看SqlSession session = sessionFactory.openSession();这行代码的实现
- public SqlSession openSession()
- {
- return openSessionFromDataSource(configuration.getDefaultExecutorType(), null, false);
- }
- private SqlSession openSessionFromDataSource(ExecutorType execType, TransactionIsolationLevel level, boolean autoCommit)
- {
- Connection connection = null;
- try
- {
- DefaultSqlSession defaultsqlsession;
- try
- {
- Environment environment = configuration.getEnvironment();
- //获取javax.sql.DataSource
- DataSource dataSource = getDataSourceFromEnvironment(environment);
- TransactionFactory transactionFactory = getTransactionFactoryFromEnvironment(environment);
- //获取java.sql.Connection
- connection = dataSource.getConnection();
- if(level != null)
- connection.setTransactionIsolation(level.getLevel());
- connection = wrapConnection(connection);
- //一系列的封装,把Connection包装成了DefaultSqlSession(SqlSession的实现类)
- org.apache.ibatis.transaction.Transaction tx = transactionFactory.newTransaction(connection, autoCommit);
- org.apache.ibatis.executor.Executor executor = configuration.newExecutor(tx, execType);
- defaultsqlsession = new DefaultSqlSession(configuration, executor, autoCommit);
- }
- catch(Exception e)
- {
- closeConnection(connection);
- throw ExceptionFactory.wrapException((new StringBuilder()).append("Error opening session. Cause: ").append(e).toString(), e);
- }
- return defaultsqlsession;
- }
- finally
- {
- ErrorContext.instance().reset();
- }
- }
4. insert数据的实现
- public int insert(String statement, Object parameter)
- {
- return update(statement, parameter);
- }
- public int update(String statement, Object parameter)
- {
- try
- {
- int i;
- try
- {
- dirty = true;
- //根据Mapper.xml配置文件中的id找到Sql语句
- org.apache.ibatis.mapping.MappedStatement ms = configuration.getMappedStatement(statement);
- //这个执行器是不是见过?构建DefaultSqlSession时传过来的
- // org.apache.ibatis.executor.Executor executor = configuration.newExecutor(tx, execType);
- i = executor.update(ms, wrapCollection(parameter));
- }
- catch(Exception e)
- {
- throw ExceptionFactory.wrapException((new StringBuilder()).append("Error updating database. Cause: ").append(e).toString(), e);
- }
- return i;
- }
- finally
- {
- ErrorContext.instance().reset();
- }
- }
a) 找一下executor,看看其update方法
i. 调用哪个executor?
- public Executor newExecutor(Transaction transaction, ExecutorType executorType)
- {
- executorType = executorType != null ? executorType : defaultExecutorType; //我们使用的是默认的Type(SIMPLE)
- executorType = executorType != null ? executorType : ExecutorType.SIMPLE;
- Executor executor;
- if(ExecutorType.BATCH == executorType)
- executor = new BatchExecutor(this, transaction);
- else
- if(ExecutorType.REUSE == executorType)
- executor = new ReuseExecutor(this, transaction);
- else
- executor = new SimpleExecutor(this, transaction);
- if(cacheEnabled)
- executor = new CachingExecutor(executor);
- executor = (Executor)interceptorChain.pluginAll(executor);
- return executor;
- }
ii. SimpleExecutor的update方法实现
- //父类BaseExecutor的update方法,调用了doUpdate方法
- public int update(MappedStatement ms, Object parameter)
- throws SQLException
- {
- ErrorContext.instance().resource(ms.getResource()).activity("executing an update").object(ms.getId());
- if(closed)
- {
- throw new ExecutorException("Executor was closed.");
- } else
- {
- clearLocalCache();
- return doUpdate(ms, parameter);
- }
- }
- //子类SimpleExecutor重写了doUpdate方法
- public int doUpdate(MappedStatement ms, Object parameter)
- throws SQLException
- {
- //java.sql.Statement
- Statement stmt = null;
- int i;
- try
- {
- Configuration configuration = ms.getConfiguration();
- StatementHandler handler = configuration.newStatementHandler(this, ms, parameter, RowBounds.DEFAULT, null);
- //大概就是拼Sql,生成对应的Statement,执行Sql,封装数据了(这块比较复杂,下次再看了)
- stmt = prepareStatement(handler);
- i = handler.update(stmt);
- }
- finally
- {
- closeStatement(stmt);
- }
- return i;
- }
看了一回源码,也感觉高端大气上档次了,下次画个图
mybatis_helloworld(2)_源码的更多相关文章
- java画图程序_图片用字母画出来_源码发布_版本二
在上一个版本:java画图程序_图片用字母画出来_源码发布 基础上,增加了图片同比例缩放,使得大像素图片可以很好地显示画在Notepad++中. 项目结构: 运行效果1: 原图:http://imag ...
- pygame系列_小球完全弹性碰撞游戏_源码下载
之前做了一个基于python的tkinter的小球完全碰撞游戏: python开发_tkinter_小球完全弹性碰撞游戏_源码下载 今天利用业余时间,写了一个功能要强大一些的小球完全碰撞游戏: 游戏名 ...
- Django_Restframwork_APIVIEW视图_源码分析
Django _VIEW视图_源码分析
- java画图程序_图片用字母画出来_源码发布
在之前写了一篇blog:java画图程序_图片用字母画出来 主要是把一些调试的截图发布出来,现在程序调试我认为可以了(当然,你如果还想调试的话,也可以下载源码自己调试). 就把源码发布出来. 项目结构 ...
- 软件包管理_rpm命令管理_yum工具管理_文件归档压缩_源码包管理
rpm命令管理软件 对于挂载的像U盘那种都会在midea目录下,但是会显示在桌面上 安装软件(i:install,v:verbose冗长的,h:human):rpm -ivh xxxx.rpm 安 ...
- bootstrap_栅格系统_响应式工具_源码分析
-----------------------------------------------------------------------------margin 为负 使盒子重叠 等高 等高 ...
- QT_文本编辑器_源码下载
源码下载: 链接: http://pan.baidu.com/s/1c21EVRy 密码: qub8 实现主要的功能有:新建,打开,保存,另存为,查找(查找的时候需要先将光标放到最下面位置才能查全,不 ...
- 『TensorFlow Internals』笔记_源码结构
零.资料集合 知乎专栏:Bob学步 知乎提问:如何高效的学习 TensorFlow 代码?. 大佬刘光聪(Github,简书) 开源书:TensorFlow Internals,强烈推荐(本博客参考书 ...
- pygame系列_font游戏字体_源码下载
在pygame游戏开发中,一个友好的UI中,漂亮的字体是少不了的 今天就给大伙带来有关pygame中字体的一些介绍说明 首先我们得判断一下我们的pygame中有没有font这个模块 if not py ...
随机推荐
- Scrum And Teamwork
Scrum Learning 概念 Scrum是迭代式增量软件开发过程,通常用于敏捷软件开发.Scrum包括了一系列实践和预定义角色的过程骨架.Scrum中的主要角色包括同项目经理类似的Scrum主管 ...
- Python新式类继承的C3算法
在Python的新式类中,方法解析顺序并非是广度优先的算法,而是采用C3算法,只是在某些情况下,C3算法的结果恰巧符合广度优先算法的结果. 可以通过代码来验证下: class NewStyleClas ...
- MySQL 优化实施方案
1.1 前言 在进行MySQL的优化之前必须要了解的就是MySQL的查询过程,很多的查询优化工作实际上就是遵循一些原则让MySQL的优化器能够按照预想的合理方式运行而已.更多关于MySQL查询相关参照 ...
- Python爬取视频(其实是一篇福利)
窗外下着小雨,作为单身程序员的我逛着逛着发现一篇好东西,来自知乎 你都用 Python 来做什么?的第一个高亮答案. 到上面去看了看,地址都是明文的,得,赶紧开始吧. 下载流式文件,requests库 ...
- 企业级memcached缓存数据库结合php使用与web管理memcached
环境 [root@cache01 ~]# cat /etc/redhat-release CentOS Linux release (Core) [root@cache01 ~]# uname -a ...
- QTableView表格自动拉伸
QTableView有四个关于自动拉伸的函数: void resizeColumnToContents(int column); void resizeColumnsToContens(); void ...
- 一句话理解字符编码(Unicode ,UTF8,UTF16)
Unicode和ASCII码属于同一级别的,都是字符集,字符集规定从1到这个字符集的最大范围每个序号都各表示什么意思.比如ASCII字符集中序号65表示"A". 那接下来的UTF8 ...
- js实现黑客帝国文字下落效果
突然想到这个效果便想实现以下. 当然免不了要百度一下,于是查找到 http://www.cnblogs.com/myvin/p/4775152.html这篇文章, 效果可以查看博文中的给出的效果图.那 ...
- 使用Java内置的Http Server构建Web应用
一.概述 使用Java技术构建Web应用时, 我们通常离不开tomcat和jetty之类的servlet容器,这些Web服务器功能强大,性能强劲,深受欢迎,是运行大型Web应用的必备神器. 虽然Jav ...
- faster-rcnn系列笔记(一)
目录: 1. 序言 2.正文 2.1 关于ROI 2.2 关于RPN 2.3 关于anchor 3. 关于数据集合制作 4. 关于参数设置 5. 参考 1.序言 叽歪一下目标检测这个模型吧,这篇笔 ...