MyBatis学习-MyBatis框架简介
MyBatis框架简介
最近因为项目需要,重新学习了一下MyBatis,发现好多都忘得差不多了,而且以前的笔记也没有了,所以还是自己去网上找的资料学习的。所以我把我所看到的,整理下来发到这里,省的以后自己再次忘记还得重新去查找。
1. MyBatis介绍
MyBatis是apache的一个开源项目iBatis,2010年这个项目由apache software foundation迁移到Google code,并且改名为MyBatis。2013年11月迁址到Github。
Mybats是一个优秀的持久层框架,它对jdbc的操作数据库的过程进行了封装,使得开发者只需要关注SQL本身,而不需要花费精力去处理例如注册驱动、创建connection、创建statement、手动设置参数、结果集检索等jdbc繁杂的过程代码。
MyBatis通过xml或注解的方式将要执行的各种statement(statement、preparedStatement、CallableStatement)配置起来,并通过java对象和statement中的sql进行映射生成最终执行的sql语句,最后由MyBatis框架执行sql并将结果映射成java对象并返回。
2. MyBatis与JDBC以及Hibernate比较
2.1.MyBatis
MyBatis是一个灵活的DAO层解决方案,满足较多的性能要求,可以在很多场合使用。
但是一下场合不建议使用:
- 需要支持多种数据库或者数据库有移植要求
- 完全动态SQL,例如:字段都要动态生成
- 使用的不是关系型数据库
2.2.JDBC
2.2.1.使用JDBC编程步骤
- 加载数据库驱动
- 创建并获取数据库链接
- 加载jdbc statement对象
- 设置sql语句
- 设置sql语句中的参数(使用preparedStatement)
- 通过statement执行sql并获取结果
- 对sql执行结果进行解析处理
- 释放资源(resultSet、preparedStatement、connection)
2.2.2.JDBC程序
public static void main(String[] args) {
Connection connection = null;
PreparedStatement preparedStatement = null;
ResultSet resultSet = null;
try {
// 加载数据库驱动
Class.forName("com.mysql.jdbc.Driver");
// 通过驱动管理类获取数据库链接
connection = DriverManager.getConnection("jdbc:mysql://localhost:3306/mybatis?characterEncoding=utf-8", "root", "root");
// 定义sql语句 ?表示占位符
String sql = "select * from user where username = ?";
// 获取预处理statement
preparedStatement = connection.prepareStatement(sql);
// 设置参数,第一个参数为sql语句中参数的序号(从1开始),第二个参数为设置的参数值
preparedStatement.setString(1, "王五");
// 向数据库发出sql执行查询,查询出结果集
resultSet = preparedStatement.executeQuery();
// 遍历查询结果集
while (resultSet.next()) {
System.out.println(resultSet.getString("id") + " " + resultSet.getString("username"));
}
} catch (Exception e) {
e.printStackTrace();
} finally {
// 释放资源
if (resultSet != null) {
try {
resultSet.close();
} catch (SQLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
if (preparedStatement != null) {
try {
preparedStatement.close();
} catch (SQLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
if (connection != null) {
try {
connection.close();
} catch (SQLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
}
此为jdbc的原生方法(未经过封装)实现了查询数据库表的操。
2.2.3.JDCB问题总结如下
- 数据库连接创建、释放频繁造成系统资源浪费,从而影响系统性能。如果使用数据库连接池可解决此问题。
- Sql语句在代码中硬编码,造成代码不易维护,实际应用中sql变化的可能较大,sql变动需要改变java代码。
- 使用preparedStatement向占位符号传递参数存在硬编码,因此sql语句的where条件不一定,可能多也可能少,修改sql还需要修改代码,系统不易维护。
- 对结果集解析存在硬编码(查询列名),sql变化导致解析代码变化,系统不易维护,如果能将数据库记录封装成pojo对象解析比较方便。
2.2.4.MyBatis解决JDBC编程问题
- 问题一:在SqlMapConfig.xml中配置数据库连接池,使用连接池管理数据库链接。
- 问题二:将Sql语句配置在XXXXmapper.xml文件中与java代码分离。
- 问题三:MyBatis自动将java对象映射至sql语句,通过statement中的parameterType定义输入参数的类型。
- 问题四:MyBatis自动将sql执行结果映射至java对象,通过statement中的resultType定义输出结果的类型。
2.3.Hibernate
- Hibernate与MyBatis不同,Hibernate是一个完全的ORM框架,而MyBatis不完全是一个ORM框架,因此程序员使用Hibernate的时候不需要自己编写sql语句,而使用MyBatis的时候程序员需要自己编写Sql语句;MyBatis可以通过XML或者注解方式灵活配置要运行的sql语句,并将java对象和sql语句映射生成最终执行的sql,最后将sql执行的结果再次映射成java对象。
- Mybatis学习门槛低,简单易学,程序员直接编写原生态sql,可严格控制sql执行性能,灵活度高,非常适合对关系数据模型要求不高的软件开发,例如互联网软件、企业运营类软件等,因为这类软件需求变化频繁,一但需求变化要求成果输出迅速。但是灵活的前提是mybatis无法做到数据库无关性,如果需要实现支持多种数据库的软件则需要自定义多套sql映射文件,工作量大。
- Hibernate对象/关系映射能力强,数据库无关性好,对于关系模型要求高的软件(例如需求固定的定制化软件)如果用hibernate开发可以节省很多代码,提高效率。但是Hibernate的学习门槛高,要精通门槛更高,而且怎么设计O/R映射,在性能和对象模型之间如何权衡,以及怎样用好Hibernate需要具有很强的经验和能力才行。
相比较MyBatis,Hibernate有以下几点:
- 学习成本:MyBatis简单易学(特别是有SQL语法基础的人),较为接近JDBC,而Hibernate因为全自动化,全部封装了,因为比较难学。
- 程序灵活性:MyBatis是半自动化,它可以直接使用SQL,灵活性高。
- 程序执行效率:MyBatis执行效率高,而Hibernate执行效率低。
- 可移植性:Hibernate较好(与数据库关联在配置中完成,HQL语句与数据库无关)
3.MyBatis架构
3.1.MyBatis结构图
- MyBatis配置:SqlMapConfig.xml,此文件作为MyBatis的全局配置文件,配置了MyBatis的运行环境等信息。mapper.xml文件即sql映射文件,文件中配置了操作数据库的sql语句。此文件需要在SqlMapConfig.xml中加载。
- 通过MyBatis环境等配置信息构造SqlSessionFactory即会话工厂。
- 由会话工厂创建SqlSession即会话,操作数据库需要通过SqlSession进行。
- MyBatis底层自定了了Executor执行器接口操作数据库,Executor接口有两个实现,一个是基本执行器,一个是缓存执行器。
- Mapped Statement也是MyBatis一个底层封装对象,它包装了MyBatis配置信息以及sql映射信息等。mapper.xml文件中一个sql对应一个Mapped Statement对象,sql的id即是Mapped Statement的id。
- Mapped Statement对sql执行输入参数进行定义,包括HasMap、基本类型、pojo,Executor通过Mapped Statement在执行sql前将输入的java对象映射至sql中,输入参数映射就是jdbc编程中对preparedStatement设置参数。
- Mapped Statement对sql执行输出结果进行定义,包括HashMap、基本类型、pojo,Executor通过Mapped Statement在执行sql后将输出结果映射至java对象中,输出结果映射过程相当于jdbc编码中对结果的编译处理过程。
3.2.名词解析
SqlSession的使用范围:
SqlSession中封装了对数据库的操作,如:查询、插入、更新、删除等;
SqlSession通过SqlSessionFactory创建;
SqlSessionFactory是通过SqlSessionFactoryBuilder进行创建。
- SqlSessionFactoryBuilder
SqlSessionFactoryBuilder用于创建SqlSessionFactory,SqlSessionFactory一旦创建完成就不需要SqlSessionFactoryBuilder了,因此SqlSession是通过SqlSessionFactory创建的。所以可以将SqlSessionFactoryBuilder当成了一个工具类使用,最佳使用范围是方法范围即方法体内部局部变量。 - SqlSessionFactory
SqlSessionFactory是一个接口,接口中定义了openSession的不同重载方法,SqlSessionFactory的最佳使用范围是整个应用运行期间,一旦创建后可能重复使用,通常以单例模式管理SqlSessionFactory。 - SqlSession
SqlSession是一个面向用户的接口,SqlSession中定义了数据库操作方法。每个线程都应该有它自己的SqlSession实例。SqlSession的实例不能共享使用,它也是线程不完全的。因此最佳的范围是请求或者方法范围,绝对不能将SqlSession实例的引用放在一个类的静态字段或者实例字段中。
打开一个 SqlSession;使用完毕就要关闭它。通常把这个关闭操作放到 finally 块中以确保每次都能执行关闭。如下:
SqlSession session = sqlSessionFactory.openSession();
try {
// do work
} finally {
session.close();
}