广告
返回顶部
首页 > 资讯 > 精选 >Mybatis SqlSessionFactory初始化原理是什么
  • 636
分享到

Mybatis SqlSessionFactory初始化原理是什么

2023-06-20 14:06:03 636人浏览 八月长安
摘要

这篇文章主要讲解了“mybatis sqlSessionFactory初始化原理是什么”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Mybatis SqlSessionFactory初始化原

这篇文章主要讲解了“mybatis sqlSessionFactory初始化原理是什么”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Mybatis SqlSessionFactory初始化原理是什么”吧!

目录
  • 引言

  • SqlSessionFactory

  • 不使用 XML 构建 SqlSessionFactory

  • SqlSessionFactoryBuilder

  • 拓展

引言

现在内卷越来越严重,关于常用的ORM框架Mybatis,小编准备了三篇文章,分别将介绍SqlSessionFactory初始化原理、SqlSession执行流程,Mybatis代理模式运行方式与最终总结,这是第一篇,感兴趣的朋友可以持续关注。

SqlSessionFactory

每个基于 MyBatis 的应用都是以一个 SqlSessionFactory 的实例为核心的。SqlSessionFactory 的实例可以通过 SqlSessionFactoryBuilder 获得。而 SqlSessionFactoryBuilder 则可以从 XML 配置文件或一个预先配置的 Configuration 实例来构建出 SqlSessionFactory 实例。

从 XML 文件中构建 SqlSessionFactory 的实例非常简单,建议使用类路径下的资源文件进行配置。 但也可以使用任意的输入流(InputStream)实例,比如用文件路径字符串或 file:// URL 构造的输入流。MyBatis 包含一个名叫 Resources 的工具类,它包含一些实用方法,使得从类路径或其它位置加载资源文件更加容易。

String resource = "org/mybatis/example/mybatis-config.xml";InputStream inputStream = Resources.getResourceAsStream(resource);SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);

XML 配置文件中包含了对 MyBatis 系统的核心设置,包括获取数据库连接实例的数据源(DataSource)以及决定事务作用域和控制方式的事务管理器(TransactionManager)。后面会再探讨 XML 配置文件的详细内容,这里先给出一个简单的示例:

<?xml version="1.0" encoding="UTF-8" ?><!DOCTYPE configuration  PUBLIC "-//mybatis.org//DTD Config 3.0//EN"  "Http://mybatis.org/dtd/mybatis-3-config.dtd"><configuration>  <environments default="development">    <environment id="development">      <transactionManager type="JDBC"/>      <dataSource type="POOLED">        <property name="driver" value="${driver}"/>        <property name="url" value="${url}"/>        <property name="username" value="${username}"/>        <property name="passWord" value="${password}"/>      </dataSource>    </environment>  </environments>  <mappers>    <mapper resource="org/mybatis/example/BlogMapper.xml"/>  </mappers></configuration>

当然,还有很多可以在 XML 文件中配置的选项,上面的示例仅罗列了最关键的部分。 注意 XML 头部的声明,它用来验证 XML 文档的正确性。environment 元素体中包含了事务管理和连接池的配置。mappers 元素则包含了一组映射器(mapper),这些映射器的 XML 映射文件包含了 SQL 代码和映射定义信息。

不使用 XML 构建 SqlSessionFactory

如果你更愿意直接从 Java 代码而不是 XML 文件中创建配置,或者想要创建你自己的配置建造器,MyBatis 也提供了完整的配置类,提供了所有与 XML 文件等价的配置项。

DataSource dataSource = BlogDataSourceFactory.getBlogDataSource();TransactionFactory transactionFactory = new JdbcTransactionFactory();Environment environment = new Environment("development", transactionFactory, dataSource);Configuration configuration = new Configuration(environment);configuration.addMapper(BlogMapper.class);SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(configuration);

注意该例中,configuration 添加了一个映射器类(mapper class)。映射器类是 Java 类,它们包含 SQL 映射注解从而避免依赖 XML 文件。不过,由于 Java 注解的一些限制以及某些 MyBatis 映射的复杂性,要使用大多数高级映射(比如:嵌套联合映射),仍然需要使用 XML 配置。有鉴于此,如果存在一个同名 XML 配置文件,MyBatis 会自动查找并加载它(在这个例子中,基于类路径和 BlogMapper.class 的类名,会加载 BlogMapper.xml)。具体细节稍后讨论。

SqlSessionFactoryBuilder

String resource = "mybatis-config.xml";InputStream inputStream = Resources.getResourceAsStream(resource);SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);

build 方法:

// 1.我们最初调用的buildpublic SqlSessionFactory build(InputStream inputStream) {    //调用了重载方法    return build(inputStream, null, null);}// 2.调用的重载方法public SqlSessionFactory build(InputStream inputStream, String environment, Properties properties) {    try {        // 创建 XMLConfigBuilder, XMLConfigBuilder是专门解析mybatis的配置文件的类        XMLConfigBuilder parser = new XMLConfigBuilder(inputStream, environment, properties);        // 执行 XML 解析        // 创建 DefaultSqlSessionFactory 对象        return build(parser.parse());    } catch (Exception e) {        //···    }}

parser.parse()

public Configuration parse() {    if (parsed) {      throw new BuilderException("Each XMLConfigBuilder can only be used once.");    }    // 标记已解析    parsed = true;    // parser.evalnode("/configuration"),    // 通过xpath 读取配置文件的节点,将读取出配置文件的所以节点    //<configuration>    //   <environments default="development">   //   </environments>    //<configuration>    parseConfiguration(parser.evalNode("/configuration"));    return configuration;  }

parseConfiguration(XNode root)

// 解析每个节点 这里每个方法进去都会有很多配置,这里就不一一解析,大家感兴趣可以看看,//  settingsElement(settings);mapperElement(root.evalNode("mappers"));private void parseConfiguration(XNode root) {    try {        //issue #117 read properties first        // 解析 <properties /> 标签        propertiesElement(root.evalNode("properties"));        // 解析 <settings /> 标签        Properties settings = settingsAsProperties(root.evalNode("settings"));        // 加载自定义的 VFS 实现类        loadCustomVfs(settings);        // 解析 <typeAliases /> 标签        typeAliasesElement(root.evalNode("typeAliases"));        // 解析 <plugins /> 标签        pluginElement(root.evalNode("plugins"));        // 解析 <objectFactory /> 标签        objectFactoryElement(root.evalNode("objectFactory"));        // 解析 <objectWrapperFactory /> 标签        objectWrapperFactoryElement(root.evalNode("objectWrapperFactory"));        // 解析 <reflectorFactory /> 标签        reflectorFactoryElement(root.evalNode("reflectorFactory"));        // 赋值 <settings /> 到 Configuration 属性        settingsElement(settings);        // read it after objectFactory and objectWrapperFactory issue #631        // 解析 <environments /> 标签        environmentsElement(root.evalNode("environments"));        // 解析 <databaseIdProvider /> 标签        databaseIdProviderElement(root.evalNode("databaseIdProvider"));        // 解析 <typeHandlers /> 标签        typeHandlerElement(root.evalNode("typeHandlers"));        // 解析 <mappers /> 标签        mapperElement(root.evalNode("mappers"));    } catch (Exception e) {        throw new BuilderException("Error parsing SQL Mapper Configuration. Cause: " + e, e);    }}    // 获取mapper  private void mapperElement(XNode parent) throws Exception {    if (parent != null) {      for (XNode child : parent.getChildren()) {         // 如果是 包将在这里进行渲染        if ("package".equals(child.getName())) {          String mapperPackage = child.getStringAttribute("name");          configuration.addMappers(mapperPackage);        } else {            // 读取resource 标签           String resource = child.getStringAttribute("resource");           // 读取url 标签            String url = child.getStringAttribute("url");            // 读取注解          String mapperClass = child.getStringAttribute("class");            // 根据不同的方式完成          if (resource != null && url == null && mapperClass == null) {            ErrorContext.instance().resource(resource);            InputStream inputStream = Resources.getResourceAsStream(resource);            XMLMapperBuilder mapperParser = new XMLMapperBuilder(inputStream, configuration, resource, configuration.getSqlFragments());            mapperParser.parse();          } else if (resource == null && url != null && mapperClass == null) {            ErrorContext.instance().resource(url);            InputStream inputStream = Resources.getUrlAsStream(url);            XMLMapperBuilder mapperParser = new XMLMapperBuilder(inputStream, configuration, url, configuration.getSqlFragments());            mapperParser.parse();          } else if (resource == null && url == null && mapperClass != null) {            Class<?> mapperInterface = Resources.classForName(mapperClass);            configuration.addMapper(mapperInterface);          } else {            throw new BuilderException("A mapper element may only specify a url, resource or class, but not more than one.");          }        }      }    }  }private void settingsElement(Properties props) {    configuration.setAutoMappingBehavior(AutoMappingBehavior.valueOf(props.getProperty("autoMappingBehavior", "PARTIAL")));    configuration.setAutoMappingUnknownColumnBehavior(AutoMappingUnknownColumnBehavior.valueOf(props.getProperty("autoMappingUnknownColumnBehavior", "NONE")));    configuration.setCacheEnabled(booleanValueOf(props.getProperty("cacheEnabled"), true));    configuration.setProxyFactory((ProxyFactory) createInstance(props.getProperty("proxyFactory"))); .....    configuration.setShrinkWhitespacesInSql(booleanValueOf(props.getProperty("shrinkWhitespacesInSql"), false));  }

mapperParser.parse();

// 这里我们先看一下 mapperParser.parse();方法 懂得原理,都是类似的  public void parse() {    if (!configuration.isResourceLoaded(resource)) {      // 加载 mapper所有子节点      configurationElement(parser.evalNode("/mapper"));      configuration.addLoadedResource(resource);        // 绑定 Namespace      bindMapperForNamespace();    } // 构建ResultMap      parsePendingResultMaps();    parsePendinGCacheRefs();    parsePendingStatements();  }   // 这里将解析整个 xml文件    private void configurationElement(XNode context) {    try {      String namespace = context.getStringAttribute("namespace");      if (namespace == null || namespace.isEmpty()) {        throw new BuilderException("Mapper's namespace cannot be empty");      }      builderAssistant.setCurrentNamespace(namespace);      cacheRefElement(context.evalNode("cache-ref"));      cacheElement(context.evalNode("cache"));      parameterMapElement(context.evalNodes("/mapper/parameterMap"));      resultMapElements(context.evalNodes("/mapper/resultMap"));      sqlElement(context.evalNodes("/mapper/sql"));        // 解析标签,      buildStatementFromContext(context.evalNodes("select|insert|update|delete"));    } catch (Exception e) {      throw new BuilderException("Error parsing Mapper XML. The XML location is '" + resource + "'. Cause: " + e, e);    }  }// 关于注解的方式的parse  public void parse() {    String resource = type.toString();    if (!configuration.isResourceLoaded(resource)) {      loadXmlResource();      configuration.addLoadedResource(resource);      assistant.setCurrentNamespace(type.getName());      parseCache();      parseCacheRef();      for (Method method : type.getMethods()) {        if (!canHaveStatement(method)) {          continue;        }        if (getAnnotationWrapper(method, false, Select.class, SelectProvider.class).isPresent()            && method.getAnnotation(ResultMap.class) == null) {          parseResultMap(method);        }        try {          parseStatement(method);        } catch (IncompleteElementException e) {          configuration.addIncompleteMethod(new MethodResolver(this, method));        }      }    }    parsePendingMethods();  }

到此Mybatis的初始化工作就完毕了,主要做了两件大事

  • 解析核心配置文件到Configuration对象,解析映射配置文件到MappedStatement对象,并保存在Configuration的对应Map中

  • 创建了DefaultSqlSessionFactory返回

通过上面的代码分析,总结了一下使用的重要的类,通过下图的装配,最终返回SqlSessionFactory,而SqlSessionFactory的最终实现是 DefaultSqlSessionFactory,关于DefaultSqlSessionFactory的介绍我们将放在下篇文章进行讲解,感兴趣的小伙伴可以持续关注!

Mybatis SqlSessionFactory初始化原理是什么

拓展

看到这里很多人就会有个疑问,这是通过配置文件的方式在进行配置,但是SpringBoot 没有这样的配置文件,是怎么做到的呢?其实springBoot是通过自定配置完成;

@Configuration// 实例化 SqlSessionFactory@ConditionalOnClass({SqlSessionFactory.class, SqlSessionFactoryBean.class})@ConditionalOnSingleCandidate(DataSource.class)// MybatisProperties 我们常用的配置@EnableConfigurationProperties({MybatisProperties.class})@AutoConfigureAfter({DataSourceAutoConfiguration.class, MybatisLanguageDriverAutoConfiguration.class})public class MybatisAutoConfiguration implements InitializingBean {}

感谢各位的阅读,以上就是“Mybatis SqlSessionFactory初始化原理是什么”的内容了,经过本文的学习后,相信大家对Mybatis SqlSessionFactory初始化原理是什么这一问题有了更深刻的体会,具体使用情况还需要大家实践验证。这里是编程网,小编将为大家推送更多相关知识点的文章,欢迎关注!

--结束END--

本文标题: Mybatis SqlSessionFactory初始化原理是什么

本文链接: https://www.lsjlt.com/news/297783.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • Mybatis SqlSessionFactory初始化原理是什么
    这篇文章主要讲解了“Mybatis SqlSessionFactory初始化原理是什么”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Mybatis SqlSessionFactory初始化原...
    99+
    2023-06-20
  • 解析Mybatis SqlSessionFactory初始化原理
    目录引言SqlSessionFactory不使用 XML 构建 SqlSessionFactorySqlSessionFactoryBuilder拓展引言 现在内卷越来越严重,关于...
    99+
    2022-11-12
  • mybatis初始化SqlSessionFactory失败的几个原因分析
    目录mybatis初始化SqlSessionFactory失败总结原因有几点SqlSessionFactory异常mybatis初始化SqlSessionFactory失败 总结原因...
    99+
    2022-11-12
  • 初始化css的原因是什么
    今天小编给大家分享一下初始化css的原因是什么的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧...
    99+
    2022-10-19
  • java什么是初始化?
    java什么是初始化?初始化就是给变量一个初始值。初始化的目的是为了让变量有值,防止使用时出现异常。【推荐学习:java课程】例如:声明一个变量:String aa = "abc"; int cc =0;其中abc和0就是初始化的值,最后值...
    99+
    2018-07-08
    java教程 java 初始化
  • java什么是初始化
    java什么是初始化初始化就是给变量一个初始值。例如:声明一个变量:String php = "www.php.cn"; int score = 100; 其中 www.php.cn 和 100 就是初始化的值,最后值不一定是 www.p...
    99+
    2016-01-15
    java基础 java 初始化
  • 什么是Java的初始化
    初始化就是给变量一个初始值。例如:声明一个变量:String aa = "abc";int cc =0;其中abc和0就是初始化的值,最后值不一定是abc或者0加入你覆盖了aa的值如下:aa = "def" ; cc=1;那最后aa变量的值...
    99+
    2015-08-08
    java入门 Java 初始化
  • 云服务器初始化失败原因是什么
    网络连接问题 在云服务器的初始化过程中,网络连接可能会成为一个重要的问题。如果网络连接不稳定或者出现问题,就会导致云服务器初始化失败。在这种情况下,我们可以采取以下措施来解决网络连接问题: (1)检查网络环境 确保云服务器所使用的网络...
    99+
    2023-10-28
    初始化 原因 服务器
  • SpringMVC初始化流程是什么
    本篇内容主要讲解“SpringMVC初始化流程是什么”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“SpringMVC初始化流程是什么”吧!框架源码是我们 Cod...
    99+
    2022-10-19
  • Git初始化命令是什么
    这篇文章给大家分享的是有关Git初始化命令是什么的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。 初始化命令命令作用git init初始化 git,创建 .git 文件感...
    99+
    2022-10-19
  • Vue初始化是什么意思
    Vue 是一款非常流行的 JavaScript 框架,广泛用于构建现代化的 Web 应用程序。在开始学习 Vue 之前,你需要先了解 Vue 的初始化过程,以便正确地使用 Vue 并创建可重复的代码。初始化 Vue 是指在页面中创建一个 V...
    99+
    2023-05-14
  • css初始化是什么意思
    这篇文章给大家分享的是有关css初始化是什么意思的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。CSS初始化是指重设浏览器的样式。不同的浏览器默认的样式可能不尽相同,所以开发时的第一件事可能就是如何把它们统一。如果...
    99+
    2023-06-14
  • spring初始化方法的执行顺序及其原理是什么
    这篇文章主要讲解了“spring初始化方法的执行顺序及其原理是什么”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“spring初始化方法的执行顺序及其原理是什么”吧!Spring中初始化方法的...
    99+
    2023-06-29
  • windows7通讯端口初始化失败是什么原因
    windows7用户在开启通讯软件的时候出现了通讯端口初始化失败的现象,这种情况是什么原因导致的呢?应当怎么办呢?之所以会有通讯端口初始化失败很有可能是大家不小心删除了注册表中的部分数据,你可以通过命令提示符窗口来解决,开启窗口以后输入ne...
    99+
    2023-07-19
  • pycharm初始化的方法是什么
    PyCharm的初始化方法是通过安装并打开PyCharm,在首次打开时,会出现一个“Welcome to PyCharm”界面,用户...
    99+
    2023-09-13
    pycharm
  • linux初始化的方法是什么
    Linux的初始化方法可以通过执行以下步骤来完成:1. 启动计算机并进入BIOS设置,在启动选项中选择从可引导介质(如硬盘或USB驱...
    99+
    2023-08-30
    linux
  • vue初始化data方法是什么
    今天小编给大家分享一下vue初始化data方法是什么的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。vue初始化data方法有...
    99+
    2023-07-04
  • 阿里云服务器初始化失败原因是什么
    1. 简介 阿里云服务器是一款强大的云计算服务,为用户提供高性能、高可靠性、高安全性的云服务器。然而,在使用阿里云服务器时,有时候会出现初始化失败的情况。那么,阿里云服务器初始化失败的原因是什么呢?本文将为您解答。2. 原因分析2.1 网络...
    99+
    2023-12-29
    阿里 初始化 原因
  • mysql初始化失败的原因及解决方法是什么
    MySQL初始化失败的原因可能有很多,常见的原因包括:1. 配置文件错误:MySQL的配置文件(my.cnf)中可能存在错误配置,比...
    99+
    2023-09-23
    mysql
  • Java类的初始化过程是什么
    Java类的初始化过程包括以下几个步骤: 加载类:当程序中使用到某个类时,Java虚拟机会先通过类加载器将该类的字节码文件加载到...
    99+
    2023-10-26
    java
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作