Mybatis简单入门 和基础知识点总结(有案例)

Mybatis介绍:

MyBatis是一个基于java的持久层框架

     MyBatis的前身叫iBatis,本是apache的一个开源项目, 2010年这个项目由apache software foundation 迁移到了google code,并且改名为MyBatis。MyBatis是支持普通SQL查询,存储过程和高级映射的优秀持久层框架。(百科查询)

我们把Mybatis的功能架构分为三层:
(1)API接口层:提供给外部使用的接口API,开发人员通过这些本地API来操纵数据库。接口层一接收到调用请求就会调用数据处理层来完成具体的数据处理。
(2)数据处理层:负责具体的SQL查找、SQL解析、SQL执行和执行结果映射处理等。它主要的目的是根据调用的请求完成一次数据库操作。
(3)基础支撑层:负责最基础的功能支撑,包括连接管理、事务管理、配置加载和缓存处理,这些都是共用的东西,将他们抽取出来作为最基础的组件。为上层的数据处理层提供最基础的支撑。


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


目录

Mybatis介绍:

JDBC存在的问题:

JDBC的编程步骤:

jdbc问题总结如下:

Mybatis架构

第一步:SqlMapConfig.xml

第二步:创建pojo

第三步:sql映射文件

第四步:加载映射文件

实例1:实现根据id查询用户

#{}和${}

parameterType和resultType

selectOne和selectList

实例二:插入用户

实例三:修改用户

实例四:删除用户

mybatis与hibernate不同

Dao层开发方法

原始Dao开发方法

Mapper动态代理开发方法

开发规范

Mapper.xml(映射文件)

Mapper小结:

SqlMapConfig.xml配置文件

properties(属性)

typeAliases(类型别名)

mappers(映射器)

JDBC存在的问题:

JDBC的编程步骤:

  1. 加载数据库驱动
  2. 创建并获取数据库链接
  3. 创建jdbc statement对象
  4. 设置sql语句
  5. 设置sql语句中的参数(使用preparedStatement)
  6. 通过statement执行sql并获取结果
  7. 对sql执行结果进行解析处理
  8. 释放资源(resultSet、preparedstatement、connection)

实例:

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的原始方法(未经封装)实现了查询数据库表记录的操作。

jdbc问题总结如下:

  1. 数据库连接创建、释放频繁造成系统资源浪费,从而影响系统性能。如果使用数据库连接池可解决此问题。
  2. Sql语句在代码中硬编码,造成代码不易维护,实际应用中sql变化的可能较大,sql变动需要改变java代码。
  3. 使用preparedStatement向占有位符号传参数存在硬编码,因为sql语句的where条件不一定,可能多也可能少,修改sql还要修改代码,系统不易维护。
  4. 对结果集解析存在硬编码(查询列名),sql变化导致解析代码变化,系统不易维护,如果能将数据库记录封装成pojo对象解析比较方便。

Mybatis架构

Mybatis简单入门 和基础知识点总结(有案例)_第1张图片

  1. mybatis配置:
  2. SqlMapConfig.xml,此文件作为mybatis的全局配置文件,配置了mybatis的运行环境等信息。mapper.xml文件即sql映射文件,文件中配置了操作数据库的sql语句。此文件需要在SqlMapConfig.xml中加载。
  3. 通过mybatis环境等配置信息构造SqlSessionFactory即会话工厂
  4. 由会话工厂创建sqlSession即会话,操作数据库需要通过sqlSession进行。
  5. mybatis底层自定义了Executor执行器接口操作数据库,Executor接口有两个实现,一个是基本执行器、一个是缓存执行器。
  6. Mapped Statement也是mybatis一个底层封装对象,它包装了mybatis配置信息及sql映射信息等。mapper.xml文件中一个sql对应一个Mapped Statement对象,sql的id即是Mapped statement的id。
  7. Mapped Statement对sql执行输入参数进行定义,包括HashMap、基本类型、pojo,Executor通过Mapped Statement在执行sql前将输入的java对象映射至sql中,输入参数映射就是jdbc编程中对preparedStatement设置参数。
  8. Mapped Statement对sql执行输出结果进行定义,包括HashMap、基本类型、pojo,Executor通过Mapped Statement在执行sql后将输出结果映射至java对象中,输出结果映射过程相当于jdbc编程中对结果的解析处理过程。

第一步:SqlMapConfig.xml

SqlMapConfig.xml是mybatis核心配置文件,配置文件内容为数据源、事务管理。




	
	·
			
                
				
                
				
                
				
				
			
		
	

第二步:创建pojo

pojo类作为mybatis进行sql映射使用,一个pojo类通常与一个数据库表对应

User.java如下:

Public class User {

   private int id;

   private String username;// 用户姓名

   private String sex;// 性别

   private Date birthday;// 生日

   private String address;// 地址

第三步:sql映射文件

创建sql映射文件User.xml:









 

第四步:加载映射文件

mybatis框架需要加载Mapper.xml映射文件

将users.xml添加在SqlMapConfig.xml,前面sqlmap/是相对路径  如下:

Mybatis简单入门 和基础知识点总结(有案例)_第2张图片

实例1:实现根据id查询用户

使用的sql:

SELECT * FROM `user` WHERE id = 1

在user.xml中添加select标签,编写sql:






	
	
	
	
	


创建一个测试类:

public class MybatisTest {
	private SqlSessionFactory sqlSessionFactory = null;

	@Before
	public void init() throws Exception {
		//1.加载核心配置文件
		String resource = "sqlMapConfig.xml";
		InputStream ras = Resources.getResourceAsStream(resource);
		
		//2.创建SqlSessionFactory
		SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(ras);

	}

	@Test
	public void testQueryUserById() throws Exception {
		// 3. 创建SqlSession对象
		SqlSession sqlSession = sqlSessionFactory.openSession();

		// 4. 执行SqlSession对象执行查询,获取结果User
		// 第一个参数是User.xml的statement的id,第二个参数是执行sql需要的参数;
        //selectOne针对只从数据库查询一条数据出来
		Object user = sqlSession.selectOne("queryUserById", 1);
        //selectOne针对只从数据库查询若干数据出来
        List list = sqlSession.selectList("queryUserByUsername1", "%王%");

		// 5. 打印结果
		System.out.println(user);

		// 6. 释放资源
		sqlSession.close();
	}
}
 
  

 

 

#{}和${}

现有sql语句 select * from user where username = 'Mike'

#{}   表示一个占位符号,当传了一个int时,username = 1,当传了一个string时,username = ‘Mike’ 自带单引号,#{}中的名字可以随便起

${}   表示拼接sql串,通过${}可以将parameterType 传入的内容拼接在sql中且不进行jdbc类型转换, 传入一个string Mike时

username = Mike 传什么样的数据都会被当成字符串拼接到sql语句中

parameterType和resultType

parameterType:指定输入参数类型,mybatis通过ognl从输入对象中获取参数值拼接在sql中。

resultType:指定输出结果类型,mybatis将sql查询结果的一行记录数据映射为resultType指定类型的对象。如果有多条数据,则分别进行映射,并把对象放到容器List中

selectOne和selectList

selectOne查询一条记录,如果使用selectOne查询多条记录则抛出异常:org.apache.ibatis.exceptions.TooManyResultsException: Expected one result (or null) to be returned by selectOne(), but found: 3  at org.apache.ibatis.session.defaults.DefaultSqlSession.selectOne(DefaultSqlSession.java:70)

selectList可以查询一条或多条记录。

实例二:插入用户

使用的sql:

INSERT INTO `user` (username,birthday,sex,address) VALUES('黄忠','2016-07-26','1','三国')

映射文件:

在User.xml配置文件中添加如下内容:   



    

       INSERT INTO `user`

       (username,birthday,sex,address) VALUES

       (#{username},#{birthday},#{sex},#{address})

    

MybatisTest中添加测试方法如下:

@Test

public void testSaveUser() {

    // 4. 创建SqlSession对象

    SqlSession sqlSession = sqlSessionFactory.openSession();



    // 5. 执行SqlSession对象执行保存

    // 创建需要保存的User

    User user = new User();

    user.setUsername("张飞");

    user.setSex("1");

    user.setBirthday(new Date());

    user.setAddress("蜀国");



    sqlSession.insert("saveUser", user);

    System.out.println(user);



    // 需要进行事务提交

    sqlSession.commit();



    // 7. 释放资源

    sqlSession.close();

}

当insert一个新用户后需要他的ID怎么办?在中添加


	
	
	
	
	
	
		SELECT LAST_INSERT_ID()
	
	INSERT INTO `user`
	(username,birthday,sex,address) VALUES
	(#{username},#{birthday},#{sex},#{address})

实例三:修改用户

配置文件中:



	UPDATE `user` SET
	username = #{username} WHERE id = #{id}

MybatisTest中添加测试方法如下:

@Test

public void testUpdateUserById() {

    // 4. 创建SqlSession对象

    SqlSession sqlSession = sqlSessionFactory.openSession();



    // 5. 执行SqlSession对象执行更新

    // 创建需要更新的User

    User user = new User();

    user.setId(26);

    user.setUsername("关羽");

    user.setSex("1");

    user.setBirthday(new Date());

    user.setAddress("蜀国");



    sqlSession.update("updateUserById", user);



    // 需要进行事务提交

    sqlSession.commit();



    // 7. 释放资源

    sqlSession.close();

}

实例四:删除用户

配置文件:


	
		delete from user where
		id=#{id}
	

测试类:

@Test
	public void testDeleteUserById() {
		// 4. 创建SqlSession对象
		SqlSession sqlSession = sqlSessionFactory.openSession();

		// 5. 执行SqlSession对象执行删除
		sqlSession.delete("deleteUserById", 48);

		// 需要进行事务提交
		sqlSession.commit();

		// 7. 释放资源
		sqlSession.close();
	}

 

mybatis与hibernate不同

Mybatis和hibernate不同,它不完全是一个ORM框架,因为MyBatis需要程序员自己编写Sql语句。mybatis可以通过XML或注解方式灵活配置要运行的sql语句,并将java对象和sql语句映射生成最终执行的sql,最后将sql执行的结果再映射生成java对象。

Mybatis学习门槛低,简单易学,程序员直接编写原生态sql,可严格控制sql执行性能,灵活度高,非常适合对关系数据模型要求不高的软件开发,例如互联网软件、企业运营类软件等,因为这类软件需求变化频繁,一但需求变化要求成果输出迅速。但是灵活的前提是mybatis无法做到数据库无关性,如果需要实现支持多种数据库的软件则需要自定义多套sql映射文件,工作量大。

Hibernate对象/关系映射能力强,数据库无关性好,对于关系模型要求高的软件(例如需求固定的定制化软件)如果用hibernate开发可以节省很多代码,提高效率。但是Hibernate的学习门槛高,要精通门槛更高,而且怎么设计O/R映射,在性能和对象模型之间如何权衡,以及怎样用好Hibernate需要具有很强的经验和能力才行。

总之,按照用户的需求在有限的资源环境下只要能做出维护性、扩展性良好的软件架构都是好架构,所以框架只有适合才是最好。

Dao层开发方法

使用MyBatis开发Dao,通常有两个方法,即原始Dao开发方法Mapper动态代理开发方法

原始Dao开发方法

原始Dao开发方法需要程序员编写Dao接口和Dao实现类。

编写映射文件:






	
	

	
	

	
	
		
			SELECT LAST_INSERT_ID()
		
		insert into user(username,birthday,sex,address)
		values(#{username},#{birthday},#{sex},#{address})
	

Dao接口

public interface UserDao {
	/**
	 * 根据id查询用户
	 * 
	 * @param id
	 * @return
	 */
	User queryUserById(int id);

	/**
	 * 根据用户名模糊查询用户
	 * 
	 * @param username
	 * @return
	 */
	List queryUserByUsername(String username);

	/**
	 * 保存用户
	 * 
	 * @param user
	 */
	void saveUser(User user);
}

Dao实现类

public class UserDaoImpl implements UserDao {
	private SqlSessionFactory sqlSessionFactory;

	public UserDaoImpl(SqlSessionFactory sqlSessionFactory) {
		super();
		this.sqlSessionFactory = sqlSessionFactory;
	}

	@Override
	public User queryUserById(int id) {
		// 创建SqlSession
		SqlSession sqlSession = this.sqlSessionFactory.openSession();
		// 执行查询逻辑
		User user = sqlSession.selectOne("queryUserById", id);
		// 释放资源
		sqlSession.close();

		return user;
	}

	@Override
	public List queryUserByUsername(String username) {
		// 创建SqlSession
		SqlSession sqlSession = this.sqlSessionFactory.openSession();

		// 执行查询逻辑
		List list = sqlSession.selectList("queryUserByUsername", username);
		// 释放资源
		sqlSession.close();
		return list;
	}

	@Override
	public void saveUser(User user) {
		// 创建SqlSession
		SqlSession sqlSession = this.sqlSessionFactory.openSession();

		// 执行保存逻辑
		sqlSession.insert("saveUser", user);
		// 提交事务
		sqlSession.commit();
		// 释放资源
		sqlSession.close();
	}
}

最后测试一下:

创建一个JUnit的测试类,对UserDao进行测试,测试代码如下:

public class UserDaoTest {
	private SqlSessionFactory sqlSessionFactory;

	@Before
	public void init() throws Exception {
		// 创建SqlSessionFactoryBuilder
		SqlSessionFactoryBuilder sqlSessionFactoryBuilder = new SqlSessionFactoryBuilder();
		// 加载SqlMapConfig.xml配置文件
		InputStream inputStream = Resources.getResourceAsStream("SqlMapConfig.xml");
		// 创建SqlsessionFactory
		this.sqlSessionFactory = sqlSessionFactoryBuilder.build(inputStream);
	}

	@Test
	public void testQueryUserById() {
		// 创建DAO
		UserDao userDao = new UserDaoImpl(this.sqlSessionFactory);
		// 执行查询
		User user = userDao.queryUserById(1);
		System.out.println(user);
	}

	@Test
	public void testQueryUserByUsername() {
		// 创建DAO

		UserDao userDao = new UserDaoImpl(this.sqlSessionFactory);
		// 执行查询
		List list = userDao.queryUserByUsername("张");
		for (User user : list) {
			System.out.println(user);
		}
	}

	@Test
	public void testSaveUser() {
		// 创建DAO
		UserDao userDao = new UserDaoImpl(this.sqlSessionFactory);

		// 创建保存对象
		User user = new User();
		user.setUsername("刘备");
		user.setBirthday(new Date());
		user.setSex("1");
		user.setAddress("蜀国");
		// 执行保存
		userDao.saveUser(user);

		System.out.println(user);
	}
}

原始Dao开发中存在以下问题:
   1. Dao方法体存在重复代码:通过SqlSessionFactory创建SqlSession,调用SqlSession的数据库操作方法
   2.调用sqlSession的数据库操作方法需要指定statement的id,这里存在硬编码,不得于开发维护。

Mapper动态代理开发方法

开发规范

Mapper接口开发方法只需要程序员编写Mapper接口(相当于Dao接口),由Mybatis框架根据接口定义创建接口的动态代理对象,代理对象的方法体同上边Dao接口实现类方法。

Mapper接口开发需要遵循以下规范:

  1. Mapper.xml文件中的namespace与mapper接口的类路径相同。
  2. Mapper接口方法名和Mapper.xml中定义的每个statement的id相同 
  3. Mapper接口方法的输入参数类型和mapper.xml中定义的每个sql 的parameterType的类型相同
  4. Mapper接口方法的输出参数类型和mapper.xml中定义的每个sql的resultType的类型相同

Mapper.xml(映射文件)

定义mapper映射文件UserMapper.xml

将UserMapper.xml放在config下mapper目录下,效果如下:

Mybatis简单入门 和基础知识点总结(有案例)_第3张图片

UserMapper.xml







	
	
	
	
	

	
	

	
	
		
			select last_insert_id()
		
		insert into user(username,birthday,sex,address) values
		(#{username},#{birthday},#{sex},#{address});
	


UserMapper接口

public interface UserMapper {
	/**
	 * 根据id查询
	 */
	User queryUserById(int id);

	/**
	 * 根据用户名查询用户
	 */
	List queryUserByUsername(String username);

	/**
	 * 保存用户
	 */
	void saveUser(User user);
}

加载UserMapper.xml文件

修改SqlMapConfig.xml文件,添加以下所示的内容:


	
		
		
	

测试方法:

public class UserMapperTest {
	private SqlSessionFactory sqlSessionFactory;

	@Before
	public void init() throws Exception {
		// 创建SqlSessionFactoryBuilder
		SqlSessionFactoryBuilder sqlSessionFactoryBuilder = new SqlSessionFactoryBuilder();
		// 加载SqlMapConfig.xml配置文件
		InputStream inputStream = Resources.getResourceAsStream("SqlMapConfig.xml");
		// 创建SqlsessionFactory
		this.sqlSessionFactory = sqlSessionFactoryBuilder.build(inputStream);
	}

	@Test
	public void testQueryUserById() {
		// 获取sqlSession,和spring整合后由spring管理
		SqlSession sqlSession = this.sqlSessionFactory.openSession();

		// 从sqlSession中获取Mapper接口的代理对象
		UserMapper userMapper = sqlSession.getMapper(UserMapper.class);
		// 执行查询方法
		User user = userMapper.queryUserById(1);
		System.out.println(user);

		// 和spring整合后由spring管理
		sqlSession.close();
	}

	@Test
	public void testQueryUserByUsername() {
		// 获取sqlSession,和spring整合后由spring管理
		SqlSession sqlSession = this.sqlSessionFactory.openSession();

		// 从sqlSession中获取Mapper接口的代理对象
		UserMapper userMapper = sqlSession.getMapper(UserMapper.class);
		// 执行查询方法
		List list = userMapper.queryUserByUsername("张");
		for (User user : list) {
			System.out.println(user);
		}

		// 和spring整合后由spring管理
		sqlSession.close();
	}

	@Test
	public void testSaveUser() {
		// 获取sqlSession,和spring整合后由spring管理
		SqlSession sqlSession = this.sqlSessionFactory.openSession();

		// 从sqlSession中获取Mapper接口的代理对象
		UserMapper userMapper = sqlSession.getMapper(UserMapper.class);
		// 创建保存对象
		User user = new User();
		user.setUsername("刘备");
		user.setBirthday(new Date());
		user.setSex("1");
		user.setAddress("蜀国");
		// 执行查询方法
		userMapper.saveUser(user);
		System.out.println(user);


		// 和spring整合后由spring管理
		sqlSession.commit();
		sqlSession.close();
	}
}

Mapper小结:

  •     selectOne和selectList

动态代理对象调用sqlSession.selectOne()和sqlSession.selectList()是根据mapper接口方法的返回值决定,如果返回list则调用selectList方法,如果返回单个对象则调用selectOne方法。

  •     namespace

mybatis官方推荐使用mapper代理方法开发mapper接口,程序员不用编写mapper接口实现类,使用mapper代理方法时,输入参数可以使用pojo包装对象或map对象,保证dao的通用性。
 

SqlMapConfig.xml配置文件

SqlMapConfig.xml中配置的内容和顺序如下:

  • properties(属性)
  • settings(全局配置参数)
  • typeAliases(类型别名)
  • typeHandlers(类型处理器)
  • objectFactory(对象工厂)
  • plugins(插件)
  • environments(环境集合属性对象)
  • environment(环境子属性对象)
  • transactionManager(事务管理)
  • dataSource(数据源)
  • mappers(映射器)

properties(属性)

在config下定义db.properties文件,如下所示:

Mybatis简单入门 和基础知识点总结(有案例)_第4张图片

db.properties配置文件内容如下

//都是key = value的形式
jdbc.driver=com.mysql.jdbc.Driver
jdbc.url=jdbc:mysql://localhost:3306/mybatis?characterEncoding=utf-8
jdbc.username=root
jdbc.password=root

SqlMapConfig.xml中引用就可以写key值名


	
	
		
		
		
		
	

	
	
		
			
			
			
			
				
				
				
				
			
		
	

	
	
		
		
	

注意: MyBatis 将按照下面的顺序来加载属性:

 在 properties 元素体内定义的属性首先被读取。 

 然后会读取properties 元素中resource或 url 加载的属性,它会覆盖已读取的同名属性。

typeAliases(类型别名)




	
	
		
		
		
	

	
		
		
		
		
		
	

	
	
		
			
			
			
			
				
				
				
				
			
		
	

	
	
		
		
	

mappers(映射器)

Mapper配置的几种方法:

使用相对于类路径的资源(现在的使用方式)

如:

使用mapper接口类路径

如:

注意:此种方法要求mapper接口名称和mapper映射文件名称相同,且放在同一个目录中

注册指定包下的所有mapper接口

如:

注意:此种方法要求mapper接口名称和mapper映射文件名称相同,且放在同一个目录中

 

 

 

你可能感兴趣的:(Mybatis)