MyBatis的使用
开源的数据持久化层框架,实体类与SQL语句之间建立映射关系。
1、使用MyBatis的开发步骤:
1.1、下载MyBatis-3.2.2.jar包并导入工程
1.2、编写MyBatis核心配置文件
1.2.1、在src目录下创建一个mybatis-config.xml的配置文件,该文件用来配置数据库连接信息和MyBatis的参数。
<?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>
<!-- 加载 database.properties 资源文件 -->
<properties resource="database.properties"></properties>
<!-- 配置mybatis的log实现为LOG4J -->
<settings>
<setting name="logImpl" value="LOG4J">
</settings>
<!-- 定义一些别名 -->
<typeAliases>
<typeAlias type="sdibt.group.entity.Book" alias="book"/>
</typeAliases>
<!-- 配置数据连接环境,可以有多个,指定默认的环境id名 -->
<environments default="development">
<!-- JDBC数据源 -->
<environment id="development">
<!-- 配置事务管理,采用JDBC的事务管理 -->
<transactionManager type="JDBC" />
<!-- POOLED:这是mybatis的自带数据源,JDNI:基于tomcat的数据源 -->
<dataSource type="POOLED">
<property name="driver" value="${mysql.driverClass}" />
<property name="url" value="${mysql.jdbcUrl}" />
<property name="username" value="${mysql.user}" />
<property name="password" value="${mysql.password}" />
</dataSource>
</environment>
</environments>
<!-- 映射文件<mapper resource="cn/cache/cache.xml"/> -->
<mappers>
<mapper resource="sdibt/group/entity/Book.xml" />
</mappers>
</configuration>
常见元素的作用如下:
(1) configuration: 配置文件的根元素节点。
(2) properties: 通过rsource属性从外部指定properties属性文件(database.properties),该属性文件描述数据库连接的相关配置(数据库驱动、连接数据库的url,数据库用户名、密码),其位置也是在/resources目录下。
(3) sttings: 设置MyBatis运行中的一些行为,比如此处设置MyBatis的log日志实现为LOG4J,即使用log4j实现日志功能。
(4) environments: 表示配置MyBatis的多套运行环境,将SQL映射到多个不同的数据库上,该元素节点下可以配置多个environment子元素节点,但是必须指定其中一个 为默认运行环境(通过defult指定)。
(5)environment:配置MyBatis的一套运行环境,需指定运行环境ID、事务管理数据源配置等相关信息。
(6)mappers:作用是告诉MyBatis去哪里找到SQL映射文件(该文件内容是开发者定义的映射sQL语句),整个项目中可以有一个或多个SQL映射文件。
(7)mapper: mappers 的子元素节点,具体指定SQL映射文件的路径,其中resource属性的值表述了SQL映射文件的路径(类资源路径)。
1.2.2、在src目录下创建一个jdbc.properties的连接数据库的配置文件,这样做的目的是以后切换数据库的时候不用修改配置文件,直接在这个目录下修改就可以。
#MySQL DataSource 数据库的配置
mysql.driverClass=com.mysql.jdbc.Driver
mysql.jdbcUrl=jdbc:mysql://127.0.0.1:3306/mybatis?characterEncoding=UTF-8
mysql.user=root
mysql.password=root
#Oracle DataSource 数据库的配置
oracle.driverClass=oracle.jdbc.driver.OracleDriver
oracle.jdbcUrl=jdbc:oracle:thin:@localhost:1521/orcl
oracle.user=sdibt
oracle.password=ok
1.3、创建实体类-POJO
1.4、DAO层–SQL映射文件
需要一个配置文件(mapper.xml),因为mybatis是需要自己创建表的,实体类可以不要(但是一般会创建一个)。在和实体类相同的目录下创建一个和实体类名称相同的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">
<mapper namespace="cn.smbms.dao.user.UserMapper">
<!-- 查询用户表记录数 -->
<select id="count" resultType="int">
<!-- 此处写数据库查询的语句 -->
select count(1) as count from smbms_user
</select>
</mapper>
**注:**如果是查询就在标签下面写SQL,插入就是insert标签,更新就是update,删除就是delete标签里面的属性 id 就是程序运行的时候根据这个id找到对应的SQL,resultType是返回值的类型,(因为查询出来的结果有很多,是一个list集合,这里的类型就是list集合里的泛型类型)。parameterType是传递的参数类型,因为可能有条件查询,还会有插入需要参数。
1.4.1、手工引入dtd文件
操作:window --> Preferences --> XML --> XMLCatalog --> Add
1.5、创建测试类:
- 读取全局配置文件
- 创建SqlSessionFactory对象,读取配置文件
- 创建SqlSession对象
- 调用mapper文件进行数据操作
public class UserMapperTest {
private Logger logger = Logger.getLogger(UserMapperTest.class);
@Before
public void setUp() throws Exception {
}
@Test
public void test() {
String resource = "mybatis-config.xml";
int count = 0;
SqlSession sqlSession = null;
try {
//1 获取mybatis-config.xml的输入流
InputStream is = Resources.getResourceAsStream(resource);
//2 创建SqlSessionFactory对象,完成对配置文件的读取
SqlSessionFactory factory = new SqlSessionFactoryBuilder().build(is);
//3 创建sqlSession
sqlSession = factory.openSession();
//4 调用mapper文件来对数据进行操作,必须先把mapper文件引入到mybatis-config.xml中
count = sqlSession.selectOne("cn.smbms.dao.user.UserMapper.count");
logger.debug("UserMapperTest count---> " + count);
} catch (IOException e) {
e.printStackTrace();
}finally{
sqlSession.close();
}
}
}
1.6、ORM-对象/关系映射
ORM(Object/Relational Mapping)即对象/关系映射是一种数据持久化技术。它在对象模型和关系型数据库之间建立起对应关系,并且提供了一种机制,通过JavaBean对象去操作数据库表中的数据。
- 0RM是一种在对象和关系数据库表两者之间进行转换的机制;
- 0RM解决方案提供在持久化对象上执行增删改查操作的API;
- 0RM解决方案提供持久化对象或属性查询的一种语言或API;
- ORM主要体现在数据访问层 。
2、MyBatis框架的优点
(1)与JDBC相比,减少了50%以上的代码量。
(2) MyBatis 是最简单的持久化框架,小巧并且简单易学。
(3) MyBatis 相当灵活,不会对应用程序或者数据库的现有设计强加任何影响,SQL写在XML里,从程序代码中彻底分离,既降低耦合度,又便于统一管理和优化,还可重用。
(4)提供XML标签,支持编写动态SQL语句。
(5)提供映射标签,支持对象与数据库的ORM字段关系映射。
3、MyBatis框架的缺点
(1) SQL语句的编写工作量较大,对开发人员编写SQL语句的功底有一定要求。
(2) SQL 语句依赖于数据库,导致数据库移植性差,不能随意更换数据库。
(3) 不支持级联更新、级联删除,使用不当,容易导致N+1的sql性能问题等(数据库1+n问题是指:查询主数据,是1次查询,查询出n条记录;根据这n条主记录,查询从记录,共需要n次,这样会带来性能问题 )。
4、MyBatis框架适用场合
Myatis专注于SQL本身,是一个足够灵活的DAO层解决方案。对性能要求很高的项目,或者需求变化较多的项目,如互联网项目,MyBais 将是不错的选择。
A: 当希望对象的持久化对应用程序完全透明时不适合使用MyBatis,完全透明应该用JDBC
B: 当数据库有移植需求或需要支持多种数据库时不适合使用MyBatis
C: 当应用程序需要完全动态的sql时不适合使用MyBatis,Mybatis只支持相对动态SQL语句,SQL写死在映射文件,不属于完全动态。Mysql因为使用SQL语句映射,不同数据库SQL语法不同,所以不适合MyBatis
5、MyBatis的基本要素
5.1、MyBatis的核心接口和类:
- SqlSessionFactoryBuilder
- –> build() --> SqlSessionFactory
- –> openSession() --> SqlSession
5.1.1、SqlSessionFactoryBuilder
1)、SqlSessionFactoryBuilder构建SqlSessionFactory的主要3个重载方法:
- build(Reader reader, String environment, Properties properties)。
- build(InpuStream nputramn, String environment, Propertis popties).
- build(Configuration config)。
配置信息以三种形式提供给SqISessionFactoryBuilder的build()方法,分别是InputStream(字节流)、Reader( 字符流)、Configuration(类),由于字节流与字符流都属于读取配置文件的方式,所以从配置信息的来源去构建一个SqlSessionFactory 有两种方式:读取XML配置文件构造方式和编程构造方式。
2)、SqlSessionFactoryBuilder的生命周期和作用域
SqlSessionFactoryBuilder的最大特点是:用过即丢。一旦创建了SqlSessionFactory对象,这个类就可以不需要存在了。因此SqlSessionFactoryBuilder的最佳适用范围就是存在于方法体内,也就是局部变量。
5.1.2、SqlSessionFactory
创建SqlSession实例:
SqlSession session=SqlSessionFactory.openSession(boolean autoCommit);
//autoCommit的值为true表示关闭事务控制,事务自动提交;false表示开启事务控制,默认为true。
最佳作用域为:application,即随着应用的生命周期一起存在;
它是单例模式:存在于整个应用运行期间,并且同时只存在一个对象实例(可将获取SqlSessionFactory的代码放在静态代码块中,保证SqlSessionFactory对象只创建一次)。
//工具类,统一创建SqlSessionFactory对象、返回SqlSession实例
public class MyBatisUtils {
private static SqlSessionFactory factory;
//放在静态代码块,保证SqlSessionFactory对象只被创建一次
static{
try {
InputStream is = Resources.getResourceAsStream("mybatis-config.xml");
factory= new SqlSessionFactoryBuilder().build(is);
} catch (IOException e) {
e.printStackTrace();
}
}
//创建
public static SqlSession createSqlSession(){
return factory.openSession(false);//(false)表示关闭事务控制
}
//关闭sqlSession
public static void closeSqlSession(SqlSession sqlSession){
if(null!=sqlSession){
sqlSession.close();
}
}
}
5.1.3、SqlSession
SqlSession包含了执行SQL所需的所有方法;对一个一次数据库会话,会话结束必须关闭(SqlSession里可以执行多次SQL语句,但一旦关闭了SqlSession,就需要再重新创建它);线程级别,不能共享,线程不安全,最佳作用域为:request或者方法体内。
SqlSession的两种使用方式:
A、通过SqlSession实例来直接执行已映射的SQL语句
@Test
public void testGetUserList() {
List<User> userList=null;
SqlSession sqlSession = null;
try {
sqlSession = MyBatisUtils.c