zoukankan      html  css  js  c++  java
  • Mybatis进阶(二)

    一.SqlSession

    1.1  SqlSession的使用范围

    SqlSession中封装了对数据库的操作,如:查询、插入、更新、删除等。

    SqlSession通过SqlSessionFactory创建

    SqlSessionFactory是通过SqlSessionFactoryBuilder进行创建。

    1.2 SqlSessionFactoryBuilder

    SqlSessionFactoryBuilder用于创建SqlSessionFacotySqlSessionFacoty一旦创建完成就不需要SqlSessionFactoryBuilder了,因为SqlSession是通过SqlSessionFactory创建的。所以可以将SqlSessionFactoryBuilder当成一个工具类使用,最佳使用范围是方法范围即方法体内局部变量。

    1.3 SqlSessionFactory

    SqlSessionFactory是一个接口,接口中定义了openSession的不同重载方法,SqlSessionFactory的最佳使用范围是整个应用运行期间,一旦创建后可以重复使用,通常以单例模式管理SqlSessionFactory

    1.4 SqlSession

    SqlSession是一个面向用户的接口sqlSession中定义了数据库操作方法。

    每个线程都应该有它自己的SqlSession实例。SqlSession的实例不能共享使用,它也是线程不安全的。因此最佳的范围是请求或方法范围。绝对不能将SqlSession实例的引用放在一个类的静态字段或实例字段中。

    打开一个 SqlSession;使用完毕就要关闭它。通常把这个关闭操作放到 finally 块中以确保每次都能执行关闭。如下:

    SqlSession session = sqlSessionFactory.openSession();
    try {
    	 // do work
    } finally {
    	session.close();
    }
    

    二.原始Dao开发

    2.1 映射文件

    <?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:命名空间,用于隔离sql,还有一个很重要的作用,后面会讲 -->
    <mapper namespace="test">
    
        <!-- 根据id查询用户 -->
        <select id="queryUserById" parameterType="int"
            resultType="cn.itcast.mybatis.pojo.User">
            select * from user where id = #{id}
        </select>
    
        <!-- 根据username模糊查询用户 -->
        <select id="queryUserByUsername" parameterType="string"
            resultType="cn.itcast.mybatis.pojo.User">
            select * from user where username like '%${value}%'
        </select>
    
        <!-- 保存用户 -->
        <insert id="saveUser" parameterType="cn.itcast.mybatis.pojo.User">
            <selectKey keyProperty="id" keyColumn="id" order="AFTER"
                resultType="int">
                SELECT LAST_INSERT_ID()
            </selectKey>
            insert into user(username,birthday,sex,address)
            values(#{username},#{birthday},#{sex},#{address})
        </insert>

    2.2 Dao接口

    先进行DAO的接口开发,编码如下:

    public interface UserDao {
    	/**
    	 * 根据id查询用户
    	 * 
    	 * @param id
    	 * @return
    	 */
    	User queryUserById(int id);
    
    	/**
    	 * 根据用户名模糊查询用户
    	 * 
    	 * @param username
    	 * @return
    	 */
    	List<User> queryUserByUsername(String username);
    
    	/**
    	 * 保存用户
    	 * 
    	 * @param user
    	 */
    	void saveUser(User user);
    }
    

    2.3. 编写Dao实现类

    public class UserDaoImpl implements UserDao {
    	private SqlSessionFactory sqlSessionFactory;
    
    	public UserDaoImpl(SqlSessionFactory sqlSessionFactory) {
    		super();
    		this.sqlSessionFactory = sqlSessionFactory;
    	}
         public User queryUserById(int id) {
    		// 创建SqlSession
    		SqlSession sqlSession = this.sqlSessionFactory.openSession();
    		// 执行查询逻辑
    		User user = sqlSession.selectOne("queryUserById", id);
    		// 释放资源
    		sqlSession.close();
    
    		return user;
    	}
    
        @Override
        public List<User> queryUserByUsername(String username) {
    		// 创建SqlSession
    		SqlSession sqlSession = this.sqlSessionFactory.openSession();
    
    		// 执行查询逻辑
    		List<User> 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();
    	}
    }
    

    2.4. Dao测试

    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<User> 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);
        }
    }

    2.5 问题

    原始Dao开发中存在以下问题:

    *Dao方法体存在重复代码:通过SqlSessionFactory创建SqlSession,调用SqlSession的数据库操作方法

    *调用sqlSession的数据库操作方法需要指定statementid,这里存在硬编码,不得于开发维护。

    三.Mapper动态代理方式

    3.1 开发规范

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

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

    1、 Mapper.xml文件中的namespacemapper接口的类路径相同。

    2、 Mapper接口方法名和Mapper.xml中定义的每个statementid相同

    3、 Mapper接口方法的输入参数类型和mapper.xml中定义的每个sql parameterType的类型相同

    4、 Mapper接口方法的输出参数类型和mapper.xml中定义的每个sqlresultType的类型相同

    3.2 Mapper.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:命名空间,用于隔离sql -->
    <!-- 还有一个很重要的作用,使用动态代理开发DAO,1. namespace必须和Mapper接口类路径一致 -->
    <mapper namespace="cn.itcast.mybatis.mapper.UserMapper">
        <!-- 根据用户id查询用户 -->
        <!-- 2. id必须和Mapper接口方法名一致 -->
        <!-- 3. parameterType必须和接口方法参数类型一致 -->
        <!-- 4. resultType必须和接口方法返回值类型一致 -->
        <select id="queryUserById" parameterType="int"
            resultType="cn.itcast.mybatis.pojo.User">
            select * from user where id = #{id}
        </select>
    
        <!-- 根据用户名查询用户 -->
        <select id="queryUserByUsername" parameterType="string"
            resultType="cn.itcast.mybatis.pojo.User">
            select * from user where username like '%${value}%'
        </select>
    
        <!-- 保存用户 -->
        <insert id="saveUser" parameterType="cn.itcast.mybatis.pojo.User">
            <selectKey keyProperty="id" keyColumn="id" order="AFTER"
                resultType="int">
                select last_insert_id()
            </selectKey>
            insert into user(username,birthday,sex,address) values
            (#{username},#{birthday},#{sex},#{address});
        </insert>
    
    </mapper>

    3.3 UserMapper(接口文件)

    public interface UserMapper {
        /**
         * 根据id查询
         * 
         * @param id
         * @return
         */
        User queryUserById(int id);
    
        /**
         * 根据用户名查询用户
         * 
         * @param username
         * @return
         */
        List<User> queryUserByUsername(String username);
    
        /**
         * 保存用户
         * 
         * @param user
         */
        void saveUser(User user);
    }

    3.4 加载UserMapper.xml文件

    <!-- 加载映射文件 -->
        <mappers>
            <mapper resource="sqlmap/User.xml" />
            <mapper resource="mapper/UserMapper.xml" />
        </mappers>

    总结:

    selectOneselectList

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

    namespace

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

    四.SqlMapConfig.xml配置文件

    4.1配置内容

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

    properties(属性)

    settings(全局配置参数)

    typeAliases(类型别名)

    typeHandlers(类型处理器)

    objectFactory(对象工厂)

    plugins(插件)

    environments(环境集合属性对象)

    environment(环境子属性对象)

    transactionManager(事务管理)

    dataSource(数据源)

    mappers(映射器)

    4.2 属性

    db.properties配置文件内容如下:

    jdbc.driver=com.mysql.jdbc.Driver
    jdbc.url=jdbc:mysql://localhost:3306/mybatis?characterEncoding=utf-8
    jdbc.username=root
    jdbc.password=root

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

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

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

    4.3 Mappers(映射器)

    Mapper配置的几种方法:

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

    如:<mapper resource="sqlmap/User.xml" />

    2.使用mapper接口类路径

    如:<mapper class="cn.itcast.mybatis.mapper.UserMapper"/>

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

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

    如:<package name="cn.itcast.mybatis.mapper"/>

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

     

  • 相关阅读:
    【BZOJ 2124】【CodeVS 1283】等差子序列
    【BZOJ 1036】【ZJOI 2008】树的统计Count
    【BZOJ 1901】【ZJU 2112】Dynamic Rankings
    【BZOJ 3924】【ZJOI 2015】幻想乡战略游戏
    【BZOJ 4103】【THUSC 2015】异或运算
    【BZOJ 4513】【SDOI 2016】储能表
    【HDU 3622】Bomb Game
    【BZOJ 3166】【HEOI 2013】Alo
    【BZOJ 3530】【SDOI 2014】数数
    【BZOJ 4567】【SCOI 2016】背单词
  • 原文地址:https://www.cnblogs.com/wangxiayun/p/9032028.html
Copyright © 2011-2022 走看看