MyBatis是第一个支持自定义SQL、存储过程和高级映射的类持久框架。MyBatis消除了大部分JDBC的样板代码、手动设置参数以及检索结果。MyBatis可以支持简单的XML和注解配置规则。使Map接口和POJO类映射到数据库字段和记录。java
下面咱们经过一个简单的项目搭建来带你认识一下MyBatis的使用和一些核心组件的讲解。mysql
为了快速构建一个MyBatis项目,咱们采用SpringBoot快速搭建的方式。搭建好后在对应的pom.xml下添加以下的maven依赖,主要做用在于引入mybatis一些jar包和类库sql
主要分为四个步骤:数据库
config.properties
数据库驱动等配置mybatis-config.xml
,引入数据库驱动,映射Mapper类<!-- mybatis 核心依赖--> <dependency> <groupId>org.mybatis</groupId> <artifactId>mybatis</artifactId> <version>3.4.6</version> </dependency> <!-- 数据库驱动包 --> <dependency> <groupId>mysql</groupId> <artifactId>mysql-connector-java</artifactId> <version>5.1.25</version> </dependency> <!-- 单元测试包--> <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>4.12</version> <scope>test</scope> </dependency>
为了便于更好的说明文章的主旨,这里就不贴出所有代码了,会贴出核心代码部分编程
编写对应的POJO类和接口式编程Mapper类,这里咱们以部门业务逻辑为例,构建一个部门类,有三个属性即部门编号、部门名称、位置,下面是部分代码:缓存
Dept.javamybatis
package com.mybatis.beans; public class Dept { private Integer deptNo; private String dname; private String loc; public Dept() {} public Dept(Integer deptNo, String dname, String loc) { this.deptNo = deptNo; this.dname = dname; this.loc = loc; } get and set... }
MyBatis最核心的功能之一就是接口式编程,它可让咱们编写Mapper接口和XML文件,从而把参数和返回结果映射到对应的字段中。架构
DeptDao.javaapp
package com.mybatis.dao; public interface DeptDao { // 经过部门名称查询 public Dept findByDname(String Dname); // 经过部门编号查询 public Dept findByDeptNo(Integer deptno); }
在/resources 下新建com.mybatis.dao 包,在其内编写对应的XML配置文件,此XML配置文件和Mapper互为映射关系。
<mapper namespace="com.mybatis.dao.DeptDao" > <sql id="DeptFindSql"> select * from dept </sql> <select id="findByDeptNo" resultType="com.mybatis.beans.Dept"> <include refid="DeptFindSql"></include> where deptno = #{deptNo} </select> <select id="findByDname" resultType="com.mybatis.beans.Dept"> <include refid="DeptFindSql"></include> where dname = #{dname} </select> </mapper>
上述的
就是映射到Mapper接口类的命名空间
<select>
标签用于编写查询语句,查询完成以后须要把结果映射到对象或者map集合等,须要用到resultType
属性指定对应的结果集。上述采用了
和 的标签写法,为了方便的映射到实体类,须要修改的话统一修改便可,下降耦合性。
构建完成基础的SQL语句和映射以后,下面来构建MySQL数据库驱动,在/resources 下建立config.properties
类
jdbc.driver=com.mysql.jdbc.Driver jdbc.url=jdbc:mysql://localhost:3306/test jdbc.username=root jdbc.password=123456
在/resources 下编写MyBatis核心配置文件myBatis-config.xml
,引入数据库驱动,映射Mapper类
<configuration> <!-- 设置导入外部properties文件位置 --> <properties resource="config.properties"></properties> <environments default="development"> <environment id="development"> <transactionManager type="JDBC"/> <dataSource type="POOLED"> <property name="driver" value="${jdbc.driver}"/> <property name="url" value="${jdbc.url}"/> <property name="username" value="${jdbc.username}"/> <property name="password" value="${jdbc.password}"/> </dataSource> </environment> </environments> <mappers> <package name="com.mybatis.dao"/> </mappers> </configuration>
configuration 标签很像是Spring 中的 beans 标签或者是基于注解的配置@Configuration,也就是MyBatis的核心配置环境,使用 properties 标签引入外部属性环境,也就是数据库驱动配置,使用 mappers 映射到Mapper所在的包,这里指的就是DeptDao.java所在的包。
在test包下面新建一个Junit单元测试类,主要流程以下:
MyBatisTest.java 代码以下:
public class MyBatisTest { private SqlSession sqlSession; /** * 读取配置文件,建立SQL工厂,打开会话 * @throws Exception */ @Before public void start() throws Exception{ InputStream is = Resources.getResourceAsStream("myBatis-config.xml"); SqlSessionFactoryBuilder builder = new SqlSessionFactoryBuilder(); SqlSessionFactory factory = builder.build(is); sqlSession = factory.openSession(); } /** * 销毁会话 */ @After public void destroy() { if(sqlSession != null){ sqlSession.close(); } } @Test public void test(){ DeptDao deptDao = sqlSession.getMapper(DeptDao.class); Dept dept = deptDao.findByDeptNo(1); System.out.println(dept.getDname()); } }
@Before 和 @After 是junit工具包中的类,@Before在执行@Test 测试其主要业务以前加载,@After 在执行@Test 测试完成以后加载。
总体结构以下:
MyBatis的架构大概是这样的,最上面是接口层,接口层就是开发人员在Mapper或者是Dao接口中的接口定义,是查询、新增、更新仍是删除操做;中间层是数据处理层,主要是配置Mapper -> xml层级之间的参数映射,SQL解析,SQL执行,结果映射的过程。上述两种流程都由基础支持层来提供功能支撑,基础支持层包括链接管理,事务管理,配置加载,缓存处理。
在不与Spring 集成的状况下,使用MyBatis执行数据库的操做主要以下:
InputStream is = Resources.getResourceAsStream("myBatis-config.xml"); SqlSessionFactoryBuilder builder = new SqlSessionFactoryBuilder(); SqlSessionFactory factory = builder.build(is); sqlSession = factory.openSession();
其中的SqlSessionFactory
,SqlSession
是MyBatis接口的核心类,尤为是SqlSession,这个接口是MyBatis中最重要的接口,这个接口可以让你执行命令,获取映射,管理事务。
在Mybatis初始化过程当中,会加载mybatis-config.xml配置文件、映射配置文件以及Mapper接口中的注解信息,解析后的配置信息会造成相应的对象并保存到Configration对象中。以后,根据该对象建立SqlSessionFactory对象。待Mybatis初始化完成后,能够经过SqlSessionFactory建立SqlSession对象并开始数据库操做。
Mybatis实现的动态SQL语句,几乎能够编写出全部知足须要的SQL。
Mybatis中scripting模块会根据用户传入的参数,解析映射文件中定义的动态SQL节点,造成数据库能执行的sql语句。
SQL语句的执行涉及多个组件,包括MyBatis的四大神器,它们是: Executor
、StatementHandler
、ParameterHandler
、ResultSetHandler
。SQL的执行过程能够
用下面这幅图来表示
MyBatis层级结构各个组件的介绍(这里只是简单介绍,具体介绍在后面):
该层保护mybatis的基础模块,它们为核心处理层提供了良好的支撑。
(1)反射模块
Mybatis中的反射模块,对Java原生的反射进行了很好的封装,提供了简易的API,方便上层调用,而且对反射操做进行了一系列的优化,好比,缓存了类的元数据(MetaClass)和对象的元数据(MetaObject),提升了反射操做的性能。
(2)类型转换模块
Mybatis的别名机制,是为了简化配置文件的,该机制是类型转换模块的主要功能之一。类型转换模块的另外一个功能是实现JDBC类型与Java类型间的转换。该功能在SQL语句绑定实参和映射查询结果集时都会涉及。在SQL语句绑定实参时,会将数据有Java类型转换成JDBC类型;在映射结果集时,会将数据有JDBC类型转换成Java类型。
(3)日志模块
Java世界里,有不少优秀的日志框架,如Log4j、Log4j二、slf4j等。Mybatis除了提供了详细的日志输出信息,还可以集成多种日志框架,其日志模块的主要功能就是集成第三方日志框架。
(4)资源加载模块
该模块主要封装了类加载器,肯定了类加载器的使用顺序,并提供了加载类文件和其它资源文件的功能。
(5) 解析器模块
该模块有两个主要功能:一个是封装了XPath,为Mybatis初始化时解析mybatis-config.xml配置文件以及映射配置文件提供支持;另外一个为处理动态SQL语句中的占位符提供支持。
(6)数据源模块
在数据源模块中,Mybatis自身提供了相应的数据源实现,也提供了与第三方数据源集成的接口。数据源是开发中的经常使用组件之一,不少开源的数据源都提供了丰富的功能,如,链接池、检测链接状态等,选择性能优秀的数据源组件,对于提供ORM框架以及整个应用的性能都是很是重要的。
(7)事务管理模块
通常地,Mybatis与Spring框架集成,由Spring框架管理事务。但Mybatis自身对数据库事务进行了抽象,提供了相应的事务接口和简单实现。
(8)缓存模块
Mybatis中有一级缓存和二级缓存,这两级缓存都依赖于缓存模块中的实现。可是,须要注意,这两级缓存与Mybatis以及整个应用是运行在同一个JVM中的,共享同一块内存,若是这两级缓存中的数据量较大,则可能影响系统中其它功能,因此须要缓存大量数据时,优先考虑使用Redis、Memcache等缓存产品。
(9)Binding模块
在调用SqlSession相应方法执行数据库操做时,须要制定映射文件中定义的SQL节点,若是sql中出现了拼写错误,那就只能在运行时才能发现。为了能尽早发现这种错误,Mybatis经过Binding模块将用户自定义的Mapper接口与映射文件关联起来,系统能够经过调用自定义Mapper接口中的方法执行相应的SQL语句完成数据库操做,从而避免上述问题。注意,在开发中,咱们只是建立了Mapper接口,而并无编写实现类,这是由于Mybatis自动为Mapper接口建立了动态代理对象。有时,自定义的Mapper接口能够彻底代替映射配置文件,但好比动态SQL语句啊等,仍是写在映射配置文件中更好。
相关参考:
mybatis的总体架构 https://my.oschina.net/liuyuantao/blog/1860807
MyBatis框架的使用及源码分析(十) CacheExecutor,SimpleExecutor,BatchExecutor ,ReuseExecutor
《深刻理解mybatis原理》 MyBatis的架构设计以及实例分析 https://blog.csdn.net/luanlouis/article/details/40422941