SqlSessionFactory 由 SqlSessionFactoryBuilder 构建而非手动 new,它解析配置生成 Configuration 后创建 DefaultSqlSessionFactory 实例;builder 一次性使用,需确保 XML 路径正确、environments 有 default 属性,且 Configuration 不可变。

MyBatis中的SqlSessionFactory是怎么创建的_工厂模式与配置解析的源码分析

SqlSessionFactory 是怎么 new 出来的?不是手动 new 的

MyBatis 不允许你直接 new SqlSessionFactory(),它必须由 SqlSessionFactoryBuilder 构建。这个 builder 本质是把配置(XML 或 Java Config)解析成内存中的 Configuration 对象,再用它实例化 DefaultSqlSessionFactory —— 这才是真正的实现类。

常见错误现象:NullPointerException 在调用 openSession() 前就抛出,往往是因为 builder 拿到的 InputStreamnull(比如 Resources.getResourceAsStream("mybatis-config.xml") 路径写错或资源没进 classpath)。

XML 配置怎么变成 Configuration 对象?靠 XPath + 自定义 NodeHandler

MyBatis 解析 mybatis-config.xml 不用 DOM/SAX,而是用 DocumentBuilder 得到原生 org.w3c.dom.Document,再用 XPath 定位节点,每个标签对应一个 NodeHandler(如 EnvironmentHandlerMappersHandler)。这不是黑盒,你可以通过继承 XMLConfigBuilder 并重写 handlerMap 来插手解析逻辑(极少数场景需要)。

容易踩的坑:<mapper> 标签的 resourceurlclass 三者互斥,同时出现会导致 BuilderException:“Duplicate property 'xxx'”。另外,typeAliases 中的别名如果和 JDK 类同名(比如起名 String),运行时不会报错,但后续映射会静默失败。

为什么 SqlSessionFactory 要设计成单例?它真线程安全吗

SqlSessionFactory 是典型的线程安全单例——它的所有方法都不修改内部状态,所有可变逻辑都委托给每次创建的 SqlSession 实例。也就是说,你可以在整个应用生命周期中只持有一个 SqlSessionFactory 实例,反复调用 openSession()

但要注意:它的线程安全性**不等于**你代码的安全性。比如你在 Spring 中误把 SqlSessionTemplate 注入为 singleton 作用域(虽然它本身是线程安全的),但若在多线程下共享同一个 SqlSession 实例(比如把它设为类字段又没加锁),就会触发 Executor 内部状态冲突,抛 ExecutorException:“Executor was closed” 或 “Transaction is already committed”。

Configuration 对象里藏了哪些关键信息?别只盯着 mappers

Configuration 是 MyBatis 的核心上下文容器,不只是 mapper 映射的集合。它还持有 mappedStatements(SQL 语句元数据)、typeAliasRegistry(别名表)、interceptorChain(插件链)、languageDriverRegistry(动态 SQL 解析器),甚至包括全局开关如 cacheEnabledlazyLoadingEnabled

最容易被忽略的一点:当你调用 configuration.addMapper(UserMapper.class) 时,MyBatis 不仅注册接口,还会扫描其所有 @Select 等注解方法,并生成对应的 MappedStatement;但如果该接口继承了其他接口(比如 BaseMapper<T>),而 BaseMapper 没被 addMapper,那么子接口里的默认方法(Java 8+)不会被识别——因为 MyBatis 只解析显式注册的接口。

真正复杂的不是创建过程,而是 Configuration 初始化时那些隐式依赖:比如 ObjectFactory 默认用 DefaultObjectFactory,但它依赖 ReflectorFactory,而后者又跟 JVM 版本、是否启用 useActualParamName 强相关——这些细节不出错时没人注意,一出就是 ReflectionException 或参数绑定为空。

本文转载于:互联网 如有侵犯,请联系zhengruancom@outlook.com删除。
免责声明:正软商城发布此文仅为传递信息,不代表正软商城认同其观点或证实其描述。