MyBatis【进阶详解】

在这里插入图片描述

1.简介

MyBatis 是一款优秀的持久层框架,它支持定制化 SQL、存储过程以及高级映射。MyBatis 避免了几乎所有的 JDBC 代码和手动设置参数以及获取结果集。MyBatis 可以使用简单的 XML 或注解来配置和映射原生信息,将接口和 Java 的 POJOs(Plain Old Java Objects,普通的 Java对象)映射成数据库中的记录

2. 背景介绍

MyBatis是支持普通SQL查询,存储过程和高级映射的优秀持久层框架。MyBatis消除了几乎所有的JDBC代码和参数的手共设置以及结果集的检索,MyBatis使用简单的XML或者注解用于配置和原始映射,将接口和Java的POJOs(Plain Ordinary Java Objects,普通的 Java对象)映射成数据库中的记录。

每个MyBatis应用程序主要利用SqlSessionFactory实例操作数据库,而SqlSessionFactory实例可以通过SqlSessionFactoryBuilder获得。SqlSessionFactoryBuilder可以从一个xml配置文件或者一个预定义的配置类实例获得。

用xml文件构建SqlSessionFactory实例是非常简单的事情,推荐在这个配置中使用类路径资源(classpath resource),但你可以使用任何Reader实例,包括用文件路径或者f://开头的url创建实例。MyBatis有一个实用类(Resources),它有很多方法,可以方便地从类路径以及其他位置加载资源。

3.框架特点

  • 简单易学:本身就很小且简单。没有任何第三方依赖,最简单安装只要两个jar文件+配置几个sql映射文件易于学习,易于使用,通过文档和源代码,可以比较完全的掌握它的设计思路和实现。
  • 灵活:mybatis不会对应用程序或者数据库的现有设计强加任何影响。 sql写在xml里,便于统一管理和优化。通过sql基本上可以实现我们不使用数据访问框架可以实现的所有功能,或许更多。
  • 解除sql与程序代码的耦合:通过提供DAL层,将业务逻辑和数据访问逻辑分离,使系统的设计更清晰,更易维护,更易单元测试。sql和代码的分离,提高了可维护性。
  • 提供映射标签,支持对象与数据库的orm字段关系映射
  • 提供对象关系映射标签,支持对象关系组建维护
  • 提供xml标签,支持编写动态sql

4. 功能架构

Mybatis的功能架构分为三层:

  • API接口层:提供给外部使用的接口API,开发人员通过这些本地API来操纵数据库。接口层一接收到调用请求就会调用数据处理层来完成具体的数据处理。

  • 数据处理层:负责具体的SQL查找、SQL解析、SQL执行和执行结果映射处理等。它主要的目的是根据调用的请求完成一次数据库操作。

  • 基础支撑层:负责最基础的功能支撑,包括连接管理、事务管理、配置加载和缓存处理,这些都是共用的东西,将他们抽取出来作为最基础的组件。为上层的数据处理层提供最基础的支撑。

    在这里插入图片描述

5. 体系结构

在这里插入图片描述

6. 使用传统的MyBatis API

​ 通过调用MyBatis中SqlSession对象的方法从而达到与数据库交互的方式,有一些类似DBUtils的操作!
在这里插入图片描述

<!-- namespace="dao接口路径"
  还可以连dao接口都不定义,只定义此mapper文件,则此处的namespace可以随意写一个“zhj”,
  执行方式sqlSession.selectList("zhj.queryAll")
         sqlSession.delete("zhj.deleteUser",1);
  基于DAO接口的开发中,底层依然是如上api
-->
<mapper namespace="zhj">
 ...
 <select id="queryAll" resultType="User">
     select id,name,gender,create_time as createTime
     from tt2
 </select>

 <!-- 删除用户 -->
 <delete id="deleteUser" parameterType="int">
     delete from tt2
     where id=#{id}
 </delete>
 ...

7.MyBatis类型解析

第一:mybatis自动处理java.util.Date

第二:mybatis自动识别1:0为true/false。数据库中存1/0,java中直接收true/false

第三:parameterType 和 resultType 解析

<--
    parameterType类型:jdk8中基本类型+String :double,float,int,short,boolean,long,date,string
 -->
<select id="queryOne" parameterType="int" resultType="User">
    select id,name,gender,create_time as createTime
    from tt2
    where id=#{id}
</select>

原理: 1>会类型字符转小写,TypeAliasRegistry初始化构造方法中注册了很多类型,先从中检查

2>如果如上没有,则用Resources.classForName(string)通过反射得到类型(核心原理是 Class.forName()类加载)

8.MyBatis事务控制

8.1 起点 :获取SqlSession对象

// ================ 起点 ===============
sqlSessionFactory.openSession();

8.2 DefaultSqlSessionFactory

// 类:DefaultSqlSessionFactory = SqlSessionFactory的实际类型
public SqlSession openSession() {
   
    // ================ openSession()方法实现 ================
    //                                                   Executor          隔离级别  自动提交
    return openSessionFromDataSource(configuration.getDefaultExecutorType(), null, false);
}
private SqlSession openSessionFromDataSource(ExecutorType execType, TransactionIsolationLevel level, boolean autoCommit) {
   
    Transaction tx = null;
    try {
   
      // 获得数据库相关配置
      final Environment environment = configuration.getEnvironment();
      final TransactionFactory transactionFactory = getTransactionFactoryFromEnvironment(environment);
      // ============== 获得事务对象,(此处需要重点关注,继续跟进 ================
      tx = transactionFactory.newTransaction(environment.getDataSource(), level, autoCommit);
      // 执行器
      final Executor executor = configuration.newExecutor(tx, execType);
      // 返回构建好的SqlSession,实际类型为DefaultSqlSession
      return new DefaultSqlSession(configuration, executor, autoCommit);
    } catch (Exception e) {
   
      closeTransaction(tx); // may have fetched a connection so lets call close()
      throw ExceptionFactory.wrapException("Error opening session.  Cause: " + e, e);
    } finally {
   
      ErrorContext.instance().reset();
    }
  }

8.3 JdbcTransactionFactory

//类:JdbcTransactionFactory
public Transaction newTransaction(DataSource ds, TransactionIsolationLevel level, boolean autoCommit) {
   
    // =============== 构建事务对象,实际类型:JdbcTransaction ==============
    return new JdbcTransaction(ds, level, autoCommit);
}

8.4 JdbcTransaction

//类:JdbcTransaction,事务对象
// 构造方法:以上传来的各项参数:【数据源+事务隔离级别+是否自动提交事务】
public JdbcTransaction(DataSource ds, TransactionIsolationLevel desiredLevel, boolean desiredAutoCommit) {
   
    // 在自己的属性中 保持了如上3项参数
    dataSource = ds;
    level = desiredLevel;
    autoCommmit = desiredAutoCommit;
}
// 事务对象中,会打开一个 Connection,其中设置了如上的参数
protected void openConnection() throws SQLException {
   
    if (log.isDebugEnabled()) {
   
      log.debug("Opening JDBC Connection");
    }
    // 1.通过数据源,获得一个Connection对象
    connection = dataSource.getConnection();
    if (level != null) {
   
      // 2.设置事务隔离级别
      connection.setTransactionIsolation(level.getLevel());
    }
    // 3.设置事务的自动提交属性,默认为false,即mybatis中默认是不自动提交事务的
    setDesiredAutoCommit(autoCommmit);
}
// 事务对象中设置 自动提交属性
protected void setDesiredAutoCommit(boolean desiredAutoCommit) {
   
	....
    connection.setAutoCommit(desiredAutoCommit);
    ....
}
// 事务对象中 提交事务
public void commit() throws SQLException {
   
    ....
    connection.commit();
	....   
}
//事务对象中 回滚事务
public void rollback() throws SQLException {
   
   	...
	connection.rollback();
    .
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值