Mybatis源码系列0-从JDBC到Mybatis

本文涉及的产品
云数据库 RDS MySQL,集群系列 2核4GB
推荐场景:
搭建个人博客
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
云数据库 RDS MySQL,高可用系列 2核4GB
简介: Mybatis源码系列0-从JDBC到Mybatis

缘起JDBC


如何使用

说Mybatis之前得先讲讲JDBC

public class JDBCTest {
    public static void main(String[] args) throws Exception {
         String url = "jdbc:mysql://XXXXX/test";
        String user = "XXXX";
        String password = "XXXX";
        try {
            // 1.加载驱动
            Class.forName("com.mysql.jdbc.Driver");
            // 2.获取连接
            Connection connection = DriverManager.getConnection(url, user, password);
            // 3.获取statement,preparedStatement
            String sql = "select * from user where id=?";
            PreparedStatement prepareStatement = connection.prepareStatement(sql);
            // 设置参数
            prepareStatement.setLong(1, 1l);
            // 4.执行查询,获取结果,
            ResultSet rs = prepareStatement.executeQuery();
            while (rs.next()) {
                System.out.println(rs.getString("userName"));
                System.out.println(rs.getString("name"));
                System.out.println(rs.getInt("age"));
            }
        } finally {
            // 5.关闭连接,释放资源
            if (rs != null) {
                rs.close();
            }
            if (prepareStatement != null) {
                prepareStatement.close();
            }
            if (connection != null) {
                connection.close();
            }
        }
    }
}


痛点分析

如果我们在一个项目中所有与数据库打交道的地方都写一堆这样这个东西,肯定是不友好的。

我们分析存在的问题:

  1. 硬编码东西,最好统一管理。我们看到第一步,二步中的驱动,数据库url,用户名,密码都是在项目启动后不变的。所以,我们应该搞一个属性配置,专门存放这些不变的东西。
  2. 获取连接: 第二步存在问题就是,我们不能在每次获取数据链接的地方,都把DriverManager.getConnection(url, user, password);写一遍吧。好的写法是对其封装一个类工厂,只在工厂类中写一遍,直接从工厂类中获取Connection
  3. 第三步:其实有两个重点:(1)sql 的分散JAVA代码中(2)sql 的参数拼接问题
  4. 第四步:就是结果的映射问题,我们通常都会定义JAVABean映射数据库字段,这种结果的处理,其实就是体力活,我们应该做一个处理器专门处理这种映射问题
  5. 第五步:链接的关闭也是一个力气活,并且如果一不小心忘了,可就不好了。所以应该做一个保证能执行完关闭的,让程序不要总是想着我关闭连接了吗?

针对这5个步骤的问题,我们自己也可以写出几个工具类出来,其实就是在造轮子了。

Mybatis 就是这么一个框架,帮助我们解决JDBC在使用上的不方便。


轮子Mybatis


(脱离Spring)使用初级阶段:DefaultSqlSession

  1. 配置
    对于硬编码的信息,将其维护到XML中,进行集中配置管理,。

mybatis-config.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>
   <!-- 1.环境配置,可以配置多个,default:指定采用哪个环境 -->
   <environments default="test">
      <!-- id:标识环境-->
      <environment id="test">
         <!-- 事务管理器,JDBC类型的事务管理器 -->
         <transactionManager type="JDBC" />
         <!-- 数据源,池类型的数据源 -->
         <dataSource type="POOLED">
            <property name="driver" value="com.mysql.jdbc.Driver" />
            <property name="url" value="jdbc:mysql://127.0.0.1:3306/test" />
            <property name="username" value="root" />
            <property name="password" value="123456" />
         </dataSource>
      </environment>
   </environments>
   <!-- 2.sql模板文件的位置的配置 -->
   <mappers>
     <mapper resource="mappers/UserMapper.xml" />
   </mappers>
</configuration>

UserMapper.xml: SQL也维护到XML中,使其从JAVA代码中脱离出来,进行集中管理。

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper
  PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
  "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<!-- mapper:根标签,namespace:命名空间,命名空间唯一 -->
<mapper namespace="UserMapper">
   <!-- 
         id:SQL语句的唯一标识,同命名空间下唯一。
         resultType:sql语句查询结果集的封装类型
    -->
   <select id="selectUser" resultType="com.wqd.model.User">
      select * from user where id= #{id}
   </select>
</mapper>
  1. 编码
// 读取配置文件
InputStream inputStream = Resources.getResourceAsStream("mybatis-config.xml");
//根据配置文件,构建sqlSessionFactory
SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
// 获取sqlSession
SqlSession sqlSession = sqlSessionFactory.openSession();
try{
//执行SQL,获取结果集
//(SQL通过命名空间+SQLID 的格式定位)
User user = sqlSession.selectOne("MyMapper.selectUser", 1);
}finally {
sqlSession .close();
}

相对于JDBC:

  • 配置集中管理,配置在xml中
  • SQL集中管理,管理在Mapper.xml中,代码中通过namespace+id的形式进行SQL的定位
  • 把连接抽象成session会话。 session的获取,统一由工厂提供,session 也做到了统一管理。
  • 参数解析,不需要了,我们只管传递参数。
  • 结果映射不用管,只管接收就完了。

将大量的编码工作,改为了配置工作。

仍然存在的可优化的点:

  1. SQL的定位仍然需要字符串的形式进行定位,不够优雅。
  2. session 的关闭仍然要自己关闭。


(脱离Spring)使用中级阶段:Mapper

为了解决SQL定位的问题,Mybatis还提供了Mapper功能。

通常,我们习惯用一个Dao来表示对数据库的访问操作。

package com.wqd.mapper
public interface UserMapper {
    public User selectUser();
}

UserMapper.xml

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper
  PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
  "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<!-- namespace 定义为接口的 全限定名 -->
<mapper namespace="com.wqd.mapper.UserMapper">
   <!-- 
         id:定义为方法名。
      resultType:sql语句查询结果集的封装类型
    -->
   <select id="selectUser" resultType="com.wqd.model.User">
      select * from user where id= #{id}
   </select>
</mapper>

编码使用

// 读取配置文件
InputStream inputStream = Resources.getResourceAsStream("mybatis-config.xml");
//根据配置文件,构建sqlSessionFactory
SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
// 获取sqlSession
SqlSession sqlSession = sqlSessionFactory.openSession();
try{
//获取Mapper ,执行操作
UserMapper userMapper = sqlSession.getMapper(UserMapper.class);
User user = userMapper.selectUser(1);
}finally{
sqlSession.close();
}

可以看出SQL的定位,通过方法调用的方式来解决了。 这样是比较符合开发人员的开发习惯的。

可优化点:

  1. session需要自己关闭。

(脱离Spring)使用高级阶段:SqlsessionManager

对于session的关闭,往往是开发人员经常遗忘的东西。有时候程序是比人靠谱的。在一些不要控制session.commit的场合,能自动关闭最好。

解决这个问题,Mybitis还提供了一个类SqlsessionManager


image.png


实现Sqlsesion

从上图我们可以看出,SqlsessionManager是Sqlsession的实现类,也就是SqlsessionManagerDefaultSqlSession 是平级的,两个都可以表示会话。

但是

// 读取配置文件
InputStream inputStream = Resources.getResourceAsStream("mybatis-config.xml");
//根据配置文件,构建sqlSessionFactory
SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
SqlSessionManager sqlSessionManager = SqlSessionManager.newInstance(sqlSessionFactory);
UserDao userMapper =sqlSessionManager.getMapper(UserDao.class);

我们看到SqlsessionManager 不需要我们手动关闭session了。是不是又轻松了许多?

其实,SqlsessionManager通过AOP技术,在执行逻辑后进行了增强,使用的开发人员不必关心connection/session的关闭问题。

当然SqlsessionManager 还解决了DefaultSqlSeesion线程不安全问题。具体SqlsessionManager 是如何实现的,在后续文章中再说。


Spring-JDBC


大框架spring,也我们准备了一个JDBC的轮子Spring-JDBC

Spring-JDBC其实就是我们常说的:JDBCTemplate

不同于mybatis的是

  • mybatis具有更加高级的操作。mybatis是一个单独的框架存在
  • JDBCTemplate 是spring的一个模块,相对功能上比较简单
  • JDBCTemplate 是可以被spring容器进行管理的。


总结


Mybatis 通过对JDBC的封装,使我们可以更加灵活的操作数据库。

接下来,我将通过源码去探索Mybatis内部一些优秀的设计,来解开Mybatis设计的秘密。


相关实践学习
如何快速连接云数据库RDS MySQL
本场景介绍如何通过阿里云数据管理服务DMS快速连接云数据库RDS MySQL,然后进行数据表的CRUD操作。
全面了解阿里云能为你做什么
阿里云在全球各地部署高效节能的绿色数据中心,利用清洁计算为万物互联的新世界提供源源不断的能源动力,目前开服的区域包括中国(华北、华东、华南、香港)、新加坡、美国(美东、美西)、欧洲、中东、澳大利亚、日本。目前阿里云的产品涵盖弹性计算、数据库、存储与CDN、分析与搜索、云通信、网络、管理与监控、应用服务、互联网中间件、移动服务、视频服务等。通过本课程,来了解阿里云能够为你的业务带来哪些帮助 &nbsp; &nbsp; 相关的阿里云产品:云服务器ECS 云服务器 ECS(Elastic Compute Service)是一种弹性可伸缩的计算服务,助您降低 IT 成本,提升运维效率,使您更专注于核心业务创新。产品详情: https://www.aliyun.com/product/ecs
相关文章
|
4月前
|
SQL XML Java
mybatis-源码深入分析(一)
mybatis-源码深入分析(一)
|
4月前
|
Java 数据库连接 数据库
SpringBoot 整合jdbc和mybatis
本文详细介绍了如何在SpringBoot项目中整合JDBC与MyBatis,并提供了具体的配置步骤和示例代码。首先,通过创建用户实体类和数据库表来准备基础环境;接着,配置Maven依赖、数据库连接及属性;最后,分别展示了JDBC与MyBatis的集成方法及其基本操作,包括增删查改等功能的实现。适合初学者快速入门。
143 3
SpringBoot 整合jdbc和mybatis
|
3月前
|
前端开发 Java 数据库连接
表白墙/留言墙 —— 中级SpringBoot项目,MyBatis技术栈MySQL数据库开发,练手项目前后端开发(带完整源码) 全方位全步骤手把手教学
本文是一份全面的表白墙/留言墙项目教程,使用SpringBoot + MyBatis技术栈和MySQL数据库开发,涵盖了项目前后端开发、数据库配置、代码实现和运行的详细步骤。
97 0
表白墙/留言墙 —— 中级SpringBoot项目,MyBatis技术栈MySQL数据库开发,练手项目前后端开发(带完整源码) 全方位全步骤手把手教学
|
3月前
|
Java 数据库连接 mybatis
Springboot整合Mybatis,MybatisPlus源码分析,自动装配实现包扫描源码
该文档详细介绍了如何在Springboot Web项目中整合Mybatis,包括添加依赖、使用`@MapperScan`注解配置包扫描路径等步骤。若未使用`@MapperScan`,系统会自动扫描加了`@Mapper`注解的接口;若使用了`@MapperScan`,则按指定路径扫描。文档还深入分析了相关源码,解释了不同情况下的扫描逻辑与优先级,帮助理解Mybatis在Springboot项目中的自动配置机制。
228 0
Springboot整合Mybatis,MybatisPlus源码分析,自动装配实现包扫描源码
|
4月前
|
SQL 存储 Java
原生JDBC简单实现Mybatis核心功能
本文介绍了在Vertx项目中使用Tdengine时,因缺乏异步JDBC驱动而采用同步驱动结合`vertx.executeBlocking`实现异步查询的方法。文中详细描述了自行开发的一个简易JDBC工具,该工具实现了SQL参数绑定与返回值映射至实体类的功能,简化了原生JDBC的繁琐操作。通过具体示例展示了其实现过程及代码细节,并与原生JDBC进行了对比,突显了其便捷性和实用性。
|
5月前
|
XML Java 数据库连接
mybatis源码研究、搭建mybatis源码运行的环境
这篇文章详细介绍了如何搭建MyBatis源码运行的环境,包括创建Maven项目、导入源码、添加代码、Debug运行研究源码,并提供了解决常见问题的方法和链接到搭建好的环境。
mybatis源码研究、搭建mybatis源码运行的环境
|
3月前
|
SQL Java 数据库连接
JDBC存在什么问题?MyBatis是如何解决的?
本文详细探讨了JDBC(Java Database Connectivity)在实际应用中存在的问题,如代码繁琐、资源管理困难、SQL语句硬编码、缺乏对象映射及事务管理和缓存支持不足等。JDBC的核心组件包括`DriverManager`、`Connection`、`Statement`、`ResultSet`和`SQLException`。为了解决这些问题,文章介绍了MyBatis框架的优势,如简化代码编写、自动资源管理、SQL与代码分离、支持对象关系映射(ORM)、简化事务管理和内置缓存机制。通过具体示例展示了MyBatis如何优化数据库操作。理解JDBC的原理有助于更好地掌握ORM框架。
46 0
|
5月前
|
Web App开发 前端开发 关系型数据库
基于SpringBoot+Vue+Redis+Mybatis的商城购物系统 【系统实现+系统源码+答辩PPT】
这篇文章介绍了一个基于SpringBoot+Vue+Redis+Mybatis技术栈开发的商城购物系统,包括系统功能、页面展示、前后端项目结构和核心代码,以及如何获取系统源码和答辩PPT的方法。
|
5月前
|
供应链 前端开发 Java
服装库存管理系统 Mybatis+Layui+MVC+JSP【完整功能介绍+实现详情+源码】
该博客文章介绍了一个使用Mybatis、Layui、MVC和JSP技术栈开发的服装库存管理系统,包括注册登录、权限管理、用户和货号管理、库存管理等功能,并提供了源码下载链接。
服装库存管理系统 Mybatis+Layui+MVC+JSP【完整功能介绍+实现详情+源码】
|
5月前
|
缓存 Java 数据库连接
我要手撕mybatis源码
该文章深入分析了MyBatis框架的初始化和数据读写阶段的源码,详细阐述了MyBatis如何通过配置文件解析、建立数据库连接、映射接口绑定、动态代理、查询缓存和结果集处理等步骤实现ORM功能,以及与传统JDBC编程相比的优势。
我要手撕mybatis源码