内容目录:

1. 动态SQL语句。

(1) Xml方式。

(2) Annotation方式。

2. MyBatis的缓存。

3. MyBatis的关联查询-多表查询。

4. MyBatis逆向工程。

1. 动态SQL语句

1.1. 动态SQL是什么

相对与固定SQL语句。根据参数不同组成不同结构的SQL语句。

这种根据参数的不同而不同的SQL语句,我们称为动态SQL语句。

1.2. 动态SQL有什么用

1.根据条件组装不同结构的SQL语句,可以提高SQL代码的重用性。

2.满足某些特定需求,如:条件判断查询。

1.3. 基于XML的实现

1.3.1. 标签包括

<sql>:用于声明SQL语句块,在操作标签中通过<include>标签引入。

<if>:类似java if(){},用于判断。

<foreach>:类似java的foreach循环,一般用于批量处理的SQL语句,如批量更新、插入、删除。

<trim>:切割标签,主要用于切割关键字的头和尾的字符。新版Mybatis的使用几率很少。

<set>:使用 set标签就是SQL语言的set关键字,可以在update 的时候,设置set 关键字后面的更新字段,逗号可以自动忽略。

<where>:使用where标签作为SQL语言的where关键字,如果where后面的条件都不成立,忽略where关键字。

<choose> <when><otherwise> : java的if else if... else。




mybatis if test byte类型比较_if mybatis or test


1.3.2. 接口文件


package org.cjw.mapper;

import org.apache.ibatis.annotations.Param;
import org.cjw.pojo.User;

import java.util.List;

public interface UserMapper {

    /**
     * 根据条件查询结果
     * @param user
     * @return
     */
    List<User> selectByCondition(User user);

    /**
     * 根据条件查询结果总数
     * @param user
     * @return
     */
    Long selectTotalByCondition(User user);

    /**
     * 修改用户
     * @param user
     * @return
     */
    int updateUserByNotNull(User user);

    /**
     * 批量删除用户
     * @param ids
     * @return
     */
    int deleteByIds(@Param("ids")Integer[] ids);

    /**
     * 批量插入
     * @param users
     * @return
     */
    int insertByBatch(@Param("users")List<User> users);
}


1.3.3. 映射文件


<?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="org.cjw.mapper.UserMapper">

    <!-- 多行查询
   resultType : 只是返回的当行数据封装的对象类型,无论单行还是多行查询都必须返回对应的实体类型User
   由于配置文件中已经设置了别名,所以这里使用别名
   -->
    <select id="selectByCondition" parameterType="User" resultType="User">
        <!--
            select * from tb_user where name like concat('%',#{name},'%') or age = #{age}
            上述SQL语句的语义来说,是一个静态SQL语句,一开始已经确定SQL的语义
            不管有没有数据,都会对全部数据进行修改,如果某一个数据没有,name会自动设置为null
            不符合实际场景

            解决方案: 使用MyBatis的动态SQL语句
         -->

        select * from tb_user
        <!--
           <include refid=""></include>
           包含引入sql片段
               refid :被引入的sql片段的id值
        -->
        <include refid="condition_sql"/>

    </select>

    <select id="selectTotalByCondition" parameterType="User" resultType="Long">
        select count(1) from tb_user
        <include refid="condition_sql"/>
    </select>

    <!--
        <sql id=""></sql>
        抽取sql片段
            id属性:片段的唯一标识,以供其他地方使用
     -->
    <sql id="condition_sql">
        <!-- 动态SQL语句
            <where>标签
                在where内部编写条件,
                1,如果只要满足一个条件<where>标签会自动拼接 WHERE 关键字拼接上对应的条件
                2,如果条件前面有 OR|AND 关键字,但是是第一个条件,那么会自动删除出这个关键字,以保证语法正确
                3,如果一个条件都没有,那么就相当于查询所有数据
             -->
        <where>
            <if test="name != '' and name != null">
                name like concat('%', #{name}, '%')
            </if>
            <if test="age != '' and age != null">
                and age = #{age}
            </if>
            <if test="email != '' and email != null">
                and email = #{email}
            </if>
            <if test="password != '' and password != null">
                and password = #{password}
            </if>
        </where>

        <!--
            另一种写法是使用trim标签:
            <trim>标签,开发者可以自定义条件,既可以指定where条件也可以指定set关键字条件

            <trim prefix="WHERE" prefixOverrides="AND | OR">
            prefix : 前缀,
                当前如果是条件就用  WERHE
                如果使用修改就用  SET
            prefixOverrides :在 WHERE 关键字后面的第一个条件,如果是 AND|OR 会自动去掉
         -->

        <!--<trim prefix="WHERE" prefixOverrides="AND|OR">
            <if test="name != '' and name != null">
                name like concat('%', #{name}, '%')
            </if>
            <if test="age != '' and name != null">
                and age = #{age}
            </if>
            <if test="email != '' and email != null">
                and email = #{email}
            </if>
            <if test="password != '' and password != null">
                and password = #{password}
            </if>
        </trim>-->
    </sql>

    <update id="updateUserByNotNull" parameterType="User">
        <!--
            update tb_user set name = #{name}, password= #{password}, age = #{age} where id = #{id}
            上述SQL语句的语义来说,是一个静态SQL语句,一开始已经确定SQL的语义
            不管有没有数据,都会对全部数据进行修改,如果某一个数据没有,name会自动设置为null
            不符合实际场景

            解决方案: 使用MyBatis的动态SQL语句,<set>标签、<trim>标签
       -->
        <!-- set标签会自动的过滤掉多余的逗号 -->
        <!--update tb_user
        <set>
            <if test="name != '' and name != null">name = #{name},</if>
            <if test="age != '' and age != null">age = #{age},</if>
            <if test="email != '' and email != null">email = #{email},</if>
            <if test="password != '' and password != null">password = #{password}</if>
        </set>
        where id = #{id}-->

        <!--
            prefix : 前缀,
           当前如果是 条件就用  WERHE
           如果使用修改 就用  SET
           prefixOverrides :如果在 WHRE 关子健 后面的第一个条件,如果是 AND|OR 会自动去掉
           suffixOverrides :如果是最后一个条件, 如果是多余的逗号(,) 会自动去掉
         -->
        update tb_user
        <trim prefix="SET" suffixOverrides=",">
            <if test="name != '' and name != null">name = #{name},</if>
            <if test="age != '' and age != null">age = #{age},</if>
            <if test="email != '' and email != null">email = #{email},</if>
            <if test="password != '' and password != null">password = #{password}</if>
        </trim>
        where id = #{id}
    </update>

    <delete id="deleteByIds" parameterType="List">
        <!-- delete from tb_user where id in (1,2,3) -->
        <!--
            <foreach collection="" open="" close="" item="" separator="">标签体内容</foreach>
                MyBatis的for循环标签
                collection:循环集合
                open:起始括号(
                close:结束括号 )
                item:集合每次循环的数据对应的变量
                separator:分割符号: (1,2,3) 数值与数值之间的分隔符 ,逗号
         -->
        delete from tb_user
        where id in
        <foreach collection="ids" item="id" open="(" close=")" separator=",">
            #{id}
        </foreach>
    </delete>

    <insert id="insertByBatch">
        insert into tb_user (name, age, email, password)
        values
        <foreach collection="users" item="user" separator=",">
            (#{user.name}, #{user.age}, #{user.email}, #{user.password})
        </foreach>
    </insert>
</mapper>


1.3.4. 测试代码


@Test
public void testSelectByCondition() {
    SqlSession session = MyBatisUtil.openSesion();
    UserMapper userMapper = session.getMapper(UserMapper.class);
    User user = new User();
    user.setName("张");
    List<User> users = userMapper.selectByCondition(user);
    for (User temp : users) {
        System.out.println(temp.getName());
    }
    session.close();
}

@Test
public void testSelectTotalByCondition() {
    SqlSession session = MyBatisUtil.openSesion();
    UserMapper userMapper = session.getMapper(UserMapper.class);
    User user = new User();
    user.setName("张");
    Long total = userMapper.selectTotalByCondition(user);
    System.out.println("total: " + total);
    session.close();
}

@Test
public void testUpdateUserByNotNull() {
    SqlSession session = MyBatisUtil.openSesion();
    try {
        UserMapper userMapper = session.getMapper(UserMapper.class);
        User user = new User();
        user.setId(1L);
        user.setName("zhangsan");
        int row = userMapper.updateUserByNotNull(user);
        System.out.println("影响的行数: " + row);
        session.commit();
    } catch (Exception e) {
        e.printStackTrace();
        session.rollback();
    } finally {
        session.close();
    }
}

@Test
public void testDeleteByIds() {
    SqlSession session = MyBatisUtil.openSesion();
    try {
        UserMapper userMapper = session.getMapper(UserMapper.class);
        Integer[] ids = {1, 2, 3, 4};
        int row = userMapper.deleteByIds(ids);
        System.out.println("影响的行数: " + row);
        session.commit();
    } catch (Exception e) {
        e.printStackTrace();
        session.rollback();
    } finally {
        session.close();
    }
}

@Test
public void testInsertByBatch() {
    SqlSession session = MyBatisUtil.openSesion();
    try {
        UserMapper userMapper = session.getMapper(UserMapper.class);
        List<User> users = new ArrayList<>();
        users.add(new User(null, "张三", 13, "zhangsan@126.com", "zhangsan123123"));
        users.add(new User(null, "张四", 14, "zhangsi@163.com", "zhangsi123123"));
        users.add(new User(null, "张五", 15, "zhangwu@qq.com", "zhangwu123123"));
        users.add(new User(null, "赵六", 16, "zhangliu@126.com", "zhaoliu123123"));
        userMapper.insertByBatch(users);
        session.commit();
    } catch (Exception e) {
        e.printStackTrace();
        session.rollback();
    } finally {
        session.close();
    }
}


1.3.5测试结果


mybatis if test byte类型比较_if mybatis or test_02


mybatis if test byte类型比较_if mybatis or test_03


mybatis if test byte类型比较_sql_04


mybatis if test byte类型比较_User_05


mybatis if test byte类型比较_SQL_06


1.4. 基于注解方式实现

动态sql除了支持xml方式以外,还支持使用纯注解的方式。

主要一下五个注解对应动态sql语句的类文件。

1. @SelectProvider

2. @InsertProvider 动态插入SQL语句对应注解。

3. @UpdateProvider 动态修改SQL语句对应注解。

4. @DeleteProvider 动态删除SQL语句对应注解。

5. @Param

1.4.1. 接口映射文件


package org.cjw.mapper;

import org.apache.ibatis.annotations.*;
import org.cjw.pojo.User;

import java.util.List;

public interface UserMapper {

    /**
     * 根据条件查询结果
     *
     * @param user
     * @return
     */
    @SelectProvider(type = UserProvider.class, method = "selectByCondition")
    List<User> selectByCondition(User user);

    /**
     * 根据条件查询结果总数
     *
     * @param user
     * @return
     */
    @SelectProvider(type = UserProvider.class, method = "selectTotalByCondition")
    Long selectTotalByCondition(User user);

    /**
     * 修改用户,参数不为空的数据才会修改
     *
     * @param user
     * @return
     */
    @UpdateProvider(type = UserProvider.class, method = "updateUserByNotNull")
    int updateUserByNotNull(User user);

    /**
     * 批量删除用户
     *
     * @param ids
     * @return
     */
    @DeleteProvider(type = UserProvider.class, method = "deleteByIds")
    int deleteByIds(@Param("ids") Integer[] ids);

    /**
     * 批量插入
     *
     * @param users
     * @return
     */
    @InsertProvider(type = UserProvider.class, method = "testInsertByBatch")
    int insertByBatch(@Param("users") List<User> users);
}


1.4.2. 动态sql语句文件


package org.cjw.mapper;

import org.apache.ibatis.annotations.*;
import org.cjw.pojo.User;

import java.util.List;

public interface UserMapper {

    /**
     * 根据条件查询结果
     *
     * @param user
     * @return
     */
    @SelectProvider(type = UserProvider.class, method = "selectByCondition")
    List<User> selectByCondition(User user);

    /**
     * 根据条件查询结果总数
     *
     * @param user
     * @return
     */
    @SelectProvider(type = UserProvider.class, method = "selectTotalByCondition")
    Long selectTotalByCondition(User user);

    /**
     * 修改用户,参数不为空的数据才会修改
     *
     * @param user
     * @return
     */
    @UpdateProvider(type = UserProvider.class, method = "updateUserByNotNull")
    int updateUserByNotNull(User user);

    /**
     * 批量删除用户
     * 在程序运行过程中已经不存在ids参数,需要如果想要使用ids这个参数中包含的数据时,需要使用@Param注解来标记参数
     * 因为在JDK1.7之前,不能通过反射获取方法参数名
     * @param ids
     * @return
     */
    @DeleteProvider(type = UserProvider.class, method = "deleteByIds")
    int deleteByIds(@Param("ids") Integer[] ids);

    /**
     * 批量插入
     * 在程序运行过程中已经不存在users参数,需要如果想要使用users这个参数中包含的数据时,需要使用@Param注解来标记参数
     * 因为在JDK1.7之前,不能通过反射获取方法参数名
     * @param users
     * @return
     */
    @InsertProvider(type = UserProvider.class, method = "testInsertByBatch")
    int insertByBatch(@Param("users") List<User> users);
}
package org.cjw.mapper;

import org.apache.ibatis.annotations.*;
import org.cjw.pojo.User;

import java.util.List;

public interface UserMapper {

    /**
     * 根据条件查询结果
     *
     * @param user
     * @return
     */
    @SelectProvider(type = UserProvider.class, method = "selectByCondition")
    List<User> selectByCondition(User user);

    /**
     * 根据条件查询结果总数
     *
     * @param user
     * @return
     */
    @SelectProvider(type = UserProvider.class, method = "selectTotalByCondition")
    Long selectTotalByCondition(User user);

    /**
     * 修改用户,参数不为空的数据才会修改
     *
     * @param user
     * @return
     */
    @UpdateProvider(type = UserProvider.class, method = "updateUserByNotNull")
    int updateUserByNotNull(User user);

    /**
     * 批量删除用户
     * 在程序运行过程中已经不存在ids参数,需要如果想要使用ids这个参数中包含的数据时,需要使用@Param注解来标记参数
     * 因为在JDK1.7之前,不能通过反射获取方法参数名
     * @param ids
     * @return
     */
    @DeleteProvider(type = UserProvider.class, method = "deleteByIds")
    int deleteByIds(@Param("ids") Integer[] ids);

    /**
     * 批量插入
     * 在程序运行过程中已经不存在users参数,需要如果想要使用users这个参数中包含的数据时,需要使用@Param注解来标记参数
     * 因为在JDK1.7之前,不能通过反射获取方法参数名
     * @param users
     * @return
     */
    @InsertProvider(type = UserProvider.class, method = "testInsertByBatch")
    int insertByBatch(@Param("users") List<User> users);
}


1.4.3. 测试代码

同上

1.4.4. 测试结果

同上

1.4.5. SQL类拼接动态sql语句

SQL类提供的API都是用于拼接SQL语句的,每一个方法对应一个子句,换句话说SQL类拼接SQL语句的单位是子句。

查询涉及的子句:select、from、join、left join、right join、where、group by、having、

order by、or、and。

更新涉及的子句:update、set、where。

插入涉及的子句:insert、values。

删除涉及的子句:delete、where。

在SQL类中对每一种子句都提供了一个对应的API。(详见Mybatis文档)

在实际开发中,95%的数据库操作都是查询操作,所以在sql语句的练习中,应该以查询sql语句为主。

1.4.6. 静态SQL语句和动态SQL语句的区别

静态SQL语句和动态SQL语句都可以通过XML配置文件和注解的形式实现。

使用xml配置文件时,需要在mybatis-config.xml文件的mappers标签中添加一个mapper标签,resource属性为配置文件路径。

使用注解时,需要在mybatis-config.xml文件的mappers标签中添加一个mapper标签,class属性值为XXXMapper映射类的全限定名。如下:


<mappers>
    <!-- xml配置文件 -->
    <mapper resource="org/cjw/mapper/UserMapper.xml"/>
    <!-- 配置类 -->
    <mapper class="org.cjw.mapper.UserProvider"/>
</mappers>


一个接口对应一个mapper标签,在开发中可能存在几十个上百个接口,现在通过mapper标签来注册接口是不现实的,所以可以使用package标签一次性注册一个包中的所有接口。如下:


<mappers>
    <!-- 配置包扫描,自动找寻xml配置文件或者注解类 -->
    <package name="org.cjw.mapper" />
</mappers>


静态SQL语句编码简单,但是灵活性不足。

动态SQL语句编码稍微复杂,但是灵活性强。

xml文件通过where、set、if、trim、sql、include标签来拼接动态SQL语句。

注解通过SQL类或原生Java代码来拼接动态SQL语句。

无论xml文件还是注解拼接动态SQL语句时,都需要使用OGNL语法来拼接,即xxx = #{XXX}的形式或者#{XXX}形式,切勿直接把参数值直接拼接到动态SQL上,否则会出现运行问题。

动态SQL语法仅仅只是拼接了SQL语句,在拼接完SQL语句后可以将其看做静态SQL语句,所以静态SQL语句的规范,动态SQL语句也需要遵循,即OGNL语法。

xml文件形式的动态SQL语句的拼接在XML文件中完成。

注解形式的动态SQL语句的拼接在一个java类的方法中完成。

2. 缓存

在Mybatis里面,所谓的缓存就是将已经查询过的记录放在内存的缓冲区或文件上,这样如果再次查询,可以通过配置的策略,命中已经查询过的记录,从而提高查询的效率。因为将数据库表中的记录读取到内存中需要经过IO,而IO操作的效率是很低,在开发中应该尽量避免IO操作,所以衍生出缓存机制。

缓存作用:提高查询的效率。

2.1. 一级缓存

Mybatis的缓存分为一级缓存、二级缓存。

一级缓存:所谓的一级缓存就是会话(SqlSesion对象)级别的缓存,同一个会话中,如果已经查询过的记录会保存一份在内存中,如果会话没有关闭,再次调用同样的查询方法,不会再查询数据库,而是直接从缓存中取出之前查询的记录(类似get请求的缓存机制)。

一级缓存默认是打开的,而且是关闭不了的。

2.1.1. 测试代码


@Test
public void testSelectByCondition() {
    SqlSession session = MyBatisUtil.openSesion();
    UserMapper userMapper = session.getMapper(UserMapper.class);
    User user = new User();
    user.setName("张");
    // 第一次查询数据库
    List<User> users = userMapper.selectByCondition(user);
    for (User temp : users) {
        System.out.println(temp.getName());
    }
    // 第二次查询数据库
    List<User> users2 = userMapper.selectByCondition(user);
    for (User temp : users2) {
        System.out.println(temp.getName());
    }
    session.close();
}


通过日志信息来判断是否有缓存。


mybatis if test byte类型比较_User_07


由图可以得知,执行了两次selectAll方法,但是只查询了一次数据库,验证了一级缓存。

如何清空一级缓存:

1.关闭会话close()。

2.进行了DML操作,提交了事务commit()。

3.手动清除缓存clearCache()。


@Test
public void testSelectByCondition() {
    SqlSession session = MyBatisUtil.openSesion();
    UserMapper userMapper = session.getMapper(UserMapper.class);
    User user = new User();
    user.setName("张");
    // 第一次查询数据库
    List<User> users = userMapper.selectByCondition(user);
    for (User temp : users) {
        System.out.println(temp.getName());
    }
    session.close();
    session = MyBatisUtil.openSesion();
    userMapper = session.getMapper(UserMapper.class);
    // 第二次查询数据库
    List<User> users2 = userMapper.selectByCondition(user);
    for (User temp : users2) {
        System.out.println(temp.getName());
    }
    session.close();
}


mybatis if test byte类型比较_sql_08


@Test
public void testSelectByCondition() {
    SqlSession session = MyBatisUtil.openSesion();
    UserMapper userMapper = session.getMapper(UserMapper.class);
    User user = new User();
    user.setName("张");
    // 第一次查询数据库
    List<User> users = userMapper.selectByCondition(user);
    for (User temp : users) {
        System.out.println(temp.getName());
    }
    Integer[] ids = {4};
    int rows = userMapper.deleteByIds(ids);
    System.out.println("影响的行数: " + rows);
    // 第二次查询数据库
    List<User> users2 = userMapper.selectByCondition(user);
    for (User temp : users2) {
        System.out.println(temp.getName());
    }
    session.close();
}


mybatis if test byte类型比较_sql_09


@Test
public void testSelectByCondition() {
    SqlSession session = MyBatisUtil.openSesion();
    UserMapper userMapper = session.getMapper(UserMapper.class);
    User user = new User();
    user.setName("张");
    // 第一次查询数据库
    List<User> users = userMapper.selectByCondition(user);
    for (User temp : users) {
        System.out.println(temp.getName());
    }
// 清除一级缓存
    session.clearCache();
    // 第二次查询数据库
    List<User> users2 = userMapper.selectByCondition(user);
    for (User temp : users2) {
        System.out.println(temp.getName());
    }
    session.close();
}


mybatis if test byte类型比较_User_10


2.2. 二级缓存

一级缓存是SqlSession对象级别,在每一次会话中有效。

二级缓存是 SqlSessionFactory级别,在整个应用都有效,可以在多个会话有效。

MyBatis本身并没有实现二级缓存二级缓存需要第三方缓存提供商的支持:

Ehcache -第三方缓存(Hibernate框架默认就是支持)

学习地址:http://www.mybatis.org/ehcache-cache/

2.2.1. 下载ehcache

https://github.com/mybatis/ehcache-cache/releases

2.2.2. 配置开启二级缓存

MyBatis开启二级缓存,新版本已经默认支持开启二级缓存。可以不改。

添加 <setting name="logImpl" value="STDOUT_LOGGING" />开启日志的配置。


<settings>
    <!-- 开启二级缓存 -->
    <setting name="cacheEnabled" value="true"/>
    <!-- 开启日志 -->
    <setting name="logImpl" value="STDOUT_LOGGING"/>
</settings>


2.2.3. 导入Ehcache.jar包


mybatis if test byte类型比较_sql_11


2.2.4. Ehcache依赖 slfj 日志框架,必须要导入slfj的两个jar包


mybatis if test byte类型比较_sql_12


2.2.5. 基于XML配置文件的二级缓存

基于XMl配置文件的时,需要使用XML配置。

【1】创建 ehcache.xml配置文件

Ehcache有自己的配置文件,在src下面创建ehcache.xml 配置文件,如下。


<ehcache>
    <!-- 缓存的磁盘位置 -->
    <diskStore path="D:/mybatis_cache"/>
    <!-- 默认的缓存策略: 如果开发者在某一个需要缓存的文件配置了自定义缓存,就不使用默认的,如果没有配置,就使用默认缓存策略-->
    <defaultCache
            maxElementsInMemory="10000"
            eternal="false"
            timeToIdleSeconds="120"
            timeToLiveSeconds="120"
            overflowToDisk="true"
    />
</ehcache>


【2】在映射文件中添加<cache>标签以及配置对应的缓存策略。

2.2.6. 缓存的命中率

命中率= 从缓存中获取数据的次数/查询的总次数

如 : 两次查询 从缓存中获取一次

0.5 = 1/2;

0.666666 = 2/3;

命中率越高缓存效果越好

因为ehcache是sqlSessionFactory级别的缓存,是针对不同的会话而言的缓存,所以如果想要测试ehcache的命中率,需要不断的关闭和开启会话进而模拟多个用户访问数据库。如果在同一个会话中执行相同的SQL语句,那么使用的mybatis自带的一级缓存,仅仅只对当前会话有效,所以会出现二级缓存的命中率一直为0.0,因为使用的一直是一级缓存。

如下:


@Test
public void testSelectByCondition() {
    for (int i = 0; i < 10; i++) {
        SqlSession session = MyBatisUtil.openSesion();
        UserMapper userMapper = session.getMapper(UserMapper.class);
        User user = new User();
        user.setName("张");
        userMapper.selectByCondition(user);
        session.close();
    }
}


测试结果:


Cache Hit Ratio [org.cjw.mapper.UserMapper]: 0.0
Opening JDBC Connection
Created connection 1472465.
Setting autocommit to false on JDBC Connection [com.mysql.jdbc.JDBC4Connection@1677d1]
==>  Preparing: select * from tb_user WHERE name like concat('%', ?, '%') 
==> Parameters: 张(String)
<==    Columns: id, name, age, email, password
<==        Row: 1, 张三, 13, zhangsan@126.com, zhangsan123123
<==        Row: 2, 张四, 14, zhangsi@163.com, zhangsi123123
<==        Row: 3, 张五, 15, zhangwu@qq.com, zhangwu123123
<==      Total: 3
DEBUG [main] - put added 0 on heap
Resetting autocommit to true on JDBC Connection [com.mysql.jdbc.JDBC4Connection@1677d1]
Closing JDBC Connection [com.mysql.jdbc.JDBC4Connection@1677d1]
Returned connection 1472465 to pool.
Cache Hit Ratio [org.cjw.mapper.UserMapper]: 0.5
Cache Hit Ratio [org.cjw.mapper.UserMapper]: 0.6666666666666666
Cache Hit Ratio [org.cjw.mapper.UserMapper]: 0.75
Cache Hit Ratio [org.cjw.mapper.UserMapper]: 0.8
Cache Hit Ratio [org.cjw.mapper.UserMapper]: 0.8333333333333334
Cache Hit Ratio [org.cjw.mapper.UserMapper]: 0.8571428571428571
Cache Hit Ratio [org.cjw.mapper.UserMapper]: 0.875
Cache Hit Ratio [org.cjw.mapper.UserMapper]: 0.8888888888888888
Cache Hit Ratio [org.cjw.mapper.UserMapper]: 0.9


从另外一方面来看,用户的每一个查询数据库的操作,都会首先经过二级缓存,如果二级缓存中没有对应的查询结果,那么就查询数据库,并将查询的记录通过一级缓存存储起来,如果还设置了二级缓存,那么也会往二级缓存中存储一份。

2.2.7. 基于注解的二级缓存

基于注解配置时,需要使用配置类。

在映射接口上打上@CacheNamespace(blocking = true)


package org.cjw.mapper;

import org.apache.ibatis.annotations.*;
import org.cjw.pojo.User;

import java.util.List;

@CacheNamespace(blocking = true)
public interface UserMapper {

    /**
     * 根据条件查询结果
     *
     * @param user
     * @return
     */
    @SelectProvider(type = UserProvider.class, method = "selectByCondition")
    List<User> selectByCondition(User user);

    /**
     * 根据条件查询结果总数
     *
     * @param user
     * @return
     */
    @SelectProvider(type = UserProvider.class, method = "selectTotalByCondition")
    Long selectTotalByCondition(User user);

    /**
     * 修改用户,参数不为空的数据才会修改
     *
     * @param user
     * @return
     */
    @UpdateProvider(type = UserProvider.class, method = "updateUserByNotNull")
    int updateUserByNotNull(User user);

    /**
     * 批量删除用户
     * 在程序运行过程中已经不存在ids参数,需要如果想要使用ids这个参数中包含的数据时,需要使用@Param注解来标记参数
     * 因为在JDK1.7之前,不能通过反射获取方法参数名
     * @param ids
     * @return
     */
    @DeleteProvider(type = UserProvider.class, method = "deleteByIds")
    int deleteByIds(@Param("ids") Integer[] ids);

    /**
     * 批量插入
     * 在程序运行过程中已经不存在users参数,需要如果想要使用users这个参数中包含的数据时,需要使用@Param注解来标记参数
     * 因为在JDK1.7之前,不能通过反射获取方法参数名
     * @param users
     * @return
     */
    @InsertProvider(type = UserProvider.class, method = "testInsertByBatch")
    int insertByBatch(@Param("users") List<User> users);
}


2.2.8. 缓存命中率


Cache Hit Ratio [org.cjw.mapper.UserMapper]: 0.0
Opening JDBC Connection
Created connection 84739718.
Setting autocommit to false on JDBC Connection [com.mysql.jdbc.JDBC4Connection@50d0686]
==>  Preparing: select * from tb_user where 1 = 1 and name like concat('%', ?,'%') 
==> Parameters: 张(String)
<==    Columns: id, name, age, email, password
<==        Row: 1, 张三, 13, zhangsan@126.com, zhangsan123123
<==        Row: 2, 张四, 14, zhangsi@163.com, zhangsi123123
<==        Row: 3, 张五, 15, zhangwu@qq.com, zhangwu123123
<==      Total: 3
Resetting autocommit to true on JDBC Connection [com.mysql.jdbc.JDBC4Connection@50d0686]
Closing JDBC Connection [com.mysql.jdbc.JDBC4Connection@50d0686]
Returned connection 84739718 to pool.
Cache Hit Ratio [org.cjw.mapper.UserMapper]: 0.5
Cache Hit Ratio [org.cjw.mapper.UserMapper]: 0.6666666666666666
Cache Hit Ratio [org.cjw.mapper.UserMapper]: 0.75
Cache Hit Ratio [org.cjw.mapper.UserMapper]: 0.8
Cache Hit Ratio [org.cjw.mapper.UserMapper]: 0.8333333333333334
Cache Hit Ratio [org.cjw.mapper.UserMapper]: 0.8571428571428571
Cache Hit Ratio [org.cjw.mapper.UserMapper]: 0.875
Cache Hit Ratio [org.cjw.mapper.UserMapper]: 0.8888888888888888
Cache Hit Ratio [org.cjw.mapper.UserMapper]: 0.9


3. MyBatis的对象关系映射(难点重点)

在实际开发中,一个业务可能可能需要查询多个数据表,而多表查询就涉及连接查询(等值连接)。

等值连接:表与表之间有一个外键关联。

我们都知道一个表对应一个POJO对象,但是对象与对象之间是没有外键关系的,对象和对象之间只有依赖关系。


对象之间关系主要是四种:
一对一 关系  one2one
一个人对应一个身份证号,一个QQ号对应一个QQ空间。
一对多 关系  one2many
    一个部门包含多个员工
多对一 关系  many2one
   多个员工属于一个部门
多对多 关系  many2many   
   多个老师对应多个学生,多个学生选择多个课程

什么关系应该从哪个对象作为中心点来看
一对多, 以one方作为中心点
 
MyBatis框架支持多表查询封装对象之间关系,主要使用collection和associatiion标签。
 
<collection> 一对多查询
<association>多对一和一对一查询


3.1. 准备多表,表之间有外键关系(员工表和部门表)


员工表
CREATE TABLE `employee` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `name` varchar(50) DEFAULT NULL,
  `dept_id` int(11) DEFAULT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=6 DEFAULT CHARSET=utf8;
部门表
CREATE TABLE `department` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `name` varchar(255) DEFAULT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=4 DEFAULT CHARSET=utf8;


3.2. 基于XML配置的关系映射

3.2.1. 多对一查询

一对一查询是多对一查询的特例,使用同一套代码。

3.2.1.1. N+1方式

N+1 : N 就是当前需要查询结果对应发送的SQL语句的条数

+1 关联查询数据需要额外多发一条SQL语句才能查询出对应的结果

3.2.1.1.1. POJO


package org.cjw.pojo;

public class Employee {

    private Long id;
    private String name;
    // 以员工为中心:多个员工对应一个部门,多对一关系,many2noe
    private Department dept;

    public Long getId() {
        return id;
    }

    public void setId(Long id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public Department getDept() {
        return dept;
    }

    public void setDept(Department dept) {
        this.dept = dept;
    }
}
package org.cjw.pojo;

public class Department {

    private Long id;
    private String name;

    public Long getId() {
        return id;
    }

    public void setId(Long id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
}


3.2.1.1.2. 映射接口


package org.cjw.mapper;

import org.cjw.pojo.Department;

public interface EmployeeMapper {

    Department selectEmployeeByPrimaryKey(Long id);
}


3.2.1.1.3. 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="org.cjw.mapper.EmployeeMapper">

 <!-- 查询主表一条sql -->
 <select id="selectEmployeeByPrimaryKey" resultMap="emp_map">
 select * from employee where id = #{emp_id}
 </select>
 
    <resultMap id="emp_map" type="Employee">
        <id property="id" column="id"/>
        <result property="name" column="name" />
 <!--
            association标签的作用:将主表某一列的值作为查询条件查询副表,并将查询结果封装成对象返回
                properties属性:最后映射的对象名
                column属性:副表查询条件
         -->
 <association property="dept" column="dept_id" select="org.cjw.mapper.EmployeeMapper.selectDeptById" />
    </resultMap>

 <!-- 查询副表N条sql -->
 <select id="selectDeptById" resultType="Department">
 select * from department where id = #{dept_id}
 </select>
</mapper>


3.2.1.1.4. 测试代码


@Test
public void testSelectEmployeeByPrimaryKey() {
    SqlSession session = MyBatisUtil.openSession();
    EmployeeMapper employeeMapper = session.getMapper(EmployeeMapper.class);
    Employee employee = employeeMapper.selectEmployeeByPrimaryKey(1L);
    System.out.println("姓名:" + employee.getName() + ", 部门:" + employee.getDept().getName());
}


3.2.1.1.5. 测试结果


姓名:张三, 部门:销售部


3.2.1.2. 等值连接查询

3.2.1.2.1. 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="org.cjw.mapper.EmployeeMapper">

 <!-- 查询主表一条sql -->
 <select id="selectEmployeeByPrimaryKey" resultMap="emp_map">
 select e.id e_id, e.name e_name, d.id d_id, d.name d_name from employee e left join department d on e.dept_id = d.id where e.id = #{emp_id}
 </select>

    <resultMap id="emp_map" type="Employee">
        <id property="id" column="e_id"/>
        <result property="name" column="e_name" />
        <collection property="dept" ofType="Department">
            <id property="id" column="d_id" />
            <result property="name" column="d_name" />
        </collection>
    </resultMap>
</mapper>


3.2.1.2.2. 测试结果


姓名:张三, 部门:销售部


当<collection>标签需要映射的记录只有一条时,默认映射为泛型类型的对象,而不是一个集合。

3.2.2. 一对多查询

一对一查询是一对多查询的特例,使用同一套代码。

3.2.2.1. N+1方式

3.2.2.1.1. POJO


package org.cjw.pojo;

public class Employee {

 private Long id;
    private String name;
    private Long deptId;

    public Long getId() {
 return id;
 }

 public void setId(Long id) {
 this.id = id;
 }

 public String getName() {
 return name;
 }

 public void setName(String name) {
 this.name = name;
 }

 public Long getDeptId() {
 return deptId;
 }

 public void setDeptId(Long deptId) {
 this.deptId = deptId;
 }
}
package org.cjw.pojo;

import java.util.List;

public class Department {

 private Long id;
    private String name;

 // 以部门为中心:一个部门对应多个与员工,一对多关系
 private List<Employee> emps;

    public Long getId() {
 return id;
 }

 public void setId(Long id) {
 this.id = id;
 }

 public String getName() {
 return name;
 }

 public void setName(String name) {
 this.name = name;
 }

 public List<Employee> getEmps() {
 return emps;
 }

 public void setEmps(List<Employee> emps) {
 this.emps = emps;
 }
}


3.2.2.1.2. 映射接口


package org.cjw.mapper;

import org.cjw.pojo.Department;

public interface DepartmentMapper {

    Department selectDeptById(Long id);
}


3.2.2.1.3. 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="org.cjw.mapper.DepartmentMapper">

    <!-- 查询主表一条sql -->
    <select id="selectDeptById" resultMap="dept_map">
        select * from department where id = #{dept_id}
    </select>

    <resultMap id="dept_map" type="Department">
        <id property="id" column="id"/>
        <result property="name" column="name" />
        <!--
            association标签:用于将主表查询结果的某列数据值作为查询副表的查询条件
                properties属性:需要映射的对象/集合
                column属性:查询条件值
                select属性:执行的查询语句
         -->
        <association property="emps" column="id" select="org.cjw.mapper.DepartmentMapper.selectEmpsByDeptId"  />
    </resultMap>

    <select id="selectEmpsByDeptId" resultType="Employee">
        select * from employee where dept_id = #{id}
    </select>

</mapper>


3.2.2.1.4. 测试代码


@Test
public void testSelectDeptById() {
    SqlSession session = MyBatisUtil.openSession();
    DepartmentMapper departmentMapper = session.getMapper(DepartmentMapper.class);
    Department dept = departmentMapper.selectDeptById(1L);
    for (Employee emp : dept.getEmps()) {
        System.out.println(emp.getName());
    }
}


3.2.2.1.5. 测试结果


张三
李四


3.2.2.2. 等值连接查询

3.2.2.2.1. 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="org.cjw.mapper.DepartmentMapper">

    <!-- 查询主表一条sql -->
    <select id="selectDeptById" resultMap="dept_map">
        select d.id d_id, d.name d_name, e.id e_id, e.name e_name from department d left join employee e on d.id = e.dept_id where d.id = #{dept_id}
    </select>

    <resultMap id="dept_map" type="Department">
        <id property="id" column="d_id"/>
        <result property="name" column="d_name" />
        <collection property="emps" column="id" ofType="Employee">
            <id property="id" column="e_id"/>
            <result property="name" column="e_name" />
            <result property="deptId" column="d_id" />
        </collection>
    </resultMap>
</mapper>


3.2.2.2.2. 测试结果


张三
李四


3.3. 基于注解配置的关系映射

3.3.1. 多对一查询

3.3.1.1. N+1方式

3.3.1.1.1. POJO


package org.cjw.pojo;

public class Employee {

    private Long id;
    private String name;
    private Department dept;

    public Long getId() {
        return id;
    }

    public void setId(Long id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public Department getDept() {
        return dept;
    }

    public void setDept(Department dept) {
        this.dept = dept;
    }
}
package org.cjw.pojo;

public class Department {

    private Long id;
    private String name;

    public Long getId() {
        return id;
    }

    public void setId(Long id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
}


3.3.1.1.2. 映射接口+注解配置


package org.cjw.mapper;

import org.apache.ibatis.annotations.*;
import org.cjw.pojo.Department;
import org.cjw.pojo.Employee;

public interface EmployeeMapper {

    @Select("select * from employee where id = #{id}")
    @Results({
            @Result(id = true, property = "id", column = "id"),
            @Result(property = "name", column = "name"),
            @Result(property = "dept", column = "dept_id", many = @Many(select = "org.cjw.mapper.EmployeeMapper.selectDeptById"))
    })
    Employee selectEmployeeByPrimaryKey(Long id);

    @Select("select * from department where id = #{dept_id}")
    Department selectDeptById(Long id);

}


3.3.1.1.3. 测试代码


@Test
public void testSelectDeptById() {
    SqlSession session = MyBatisUtil.openSession();
    EmployeeMapper employeeMapper = session.getMapper(EmployeeMapper.class);
    Employee employee = employeeMapper.selectEmployeeByPrimaryKey(1L);
    System.out.println("姓名:" + employee.getName() + ",部门:" + employee.getDept().getName());
}


3.3.1.1.4. 测试结果


姓名:张三,部门:销售部


3.3.1.2. 等值连接查询

连接查询暂时未发现解决方案,有知道的朋友可以评论留言告诉我。

3.3.2. 一对多查询

3.3.2.0.1. POJO


package org.cjw.pojo;

public class Employee {

    private Long id;
    private String name;
    private Long dept_id;

    public Long getId() {
        return id;
    }

    public void setId(Long id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public Long getDept_id() {
        return dept_id;
    }

    public void setDept_id(Long dept_id) {
        this.dept_id = dept_id;
    }
}
package org.cjw.pojo;

import java.util.List;

public class Department {

    private Long id;
    private String name;
    private List<Employee> emps;

    public Long getId() {
        return id;
    }

    public void setId(Long id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public List<Employee> getEmps() {
        return emps;
    }

    public void setEmps(List<Employee> emps) {
        this.emps = emps;
    }
}


3.3.2.0.2. 映射接口+注解配置


package org.cjw.mapper;

import org.apache.ibatis.annotations.Many;
import org.apache.ibatis.annotations.Result;
import org.apache.ibatis.annotations.Results;
import org.apache.ibatis.annotations.Select;
import org.cjw.pojo.Department;
import org.cjw.pojo.Employee;

import java.util.List;

public interface DepartmentMapper {

    @Select("select * from department where id = #{dept_id}")
    @Results({
            @Result(id = true, property = "id", column = "id"),
            @Result(property = "name", column = "name"),
            @Result(property = "emps", column = "id", many = @Many(select = "org.cjw.mapper.DepartmentMapper.selectEmpsByDeptId"))
    })
    Department selectDeptById(Long id);

    @Select("select * from employee where dept_id = #{dept_id}")
    List<Employee> selectEmpsByDeptId(Long id);
}


3.3.2.0.3. 测试代码


@Test
public void testSelectDeptById() {
    SqlSession session = MyBatisUtil.openSession();
    DepartmentMapper departmentMapper = session.getMapper(DepartmentMapper.class);
    Department dept = departmentMapper.selectDeptById(1L);
    for (Employee emp : dept.getEmps()) {
        System.out.println("部门:" + dept.getName() + ",雇员名:" + emp.getName());
    }
}


3.3.2.0.4. 测试结果


部门:销售部,雇员名:张三
部门:销售部,雇员名:李四


3.3.2.1. 等值连接查询

同上,有知道的小伙伴可以留言给我。

3.4. 对象管理映射个人理解

在开发中,一般pojo中的常见属性除了基本类型、包装类型、String之外,还可能出现对象属性、集合属性,对于基本、包装、String类型,可以直接进行映射,但是对象属性、集合属性并不能,因为内部还需要再指定字段和属性的映射关系。换言之,对象属性、集合属性本身就需要指定映射关系,而其又属于一个pojo对象,这个pojo对象本身也需要指定映射关系,因此对于存在pojo对象属性和集合类型属性的pojo对象的映射,咱们需要使用N+1策略或者等值连接策略。

N+1策略则是先把所有pojo对象查询出来,然后再根据外键字段去查询对象属性或者集合属性,使用association标签(property、column、select标签属性)。

等值连接策略则是通过连表查询,将pojo对象的属性相关信息都查询出来,然后在内存中根据指定的映射关系将字段值赋值到属性上,对于pojo对象属性和集合类型属性,使用collection标签(property、ofType属性)来指定字段值和其属性的映射关系。

N+1策略操作数据库次数为记录数+1次,等值连接策略操作数据库次数为1次。

4. MyBatis的逆向工程

MyBatis的逆向工程能自动帮开发者生成数据库表对应的pojo实体文件,自动生成映射文件

自动生成表的各种(CRUD)的sql语句,但是只能做单表操作,联合查询还得开发者自己改造。

使用逆向工程得先在Eclipse安装逆向工程的插件。

4.1. 插件安装步骤


mybatis if test byte类型比较_if mybatis or test_13


mybatis if test byte类型比较_User_14


mybatis if test byte类型比较_SQL_15


判断是否安装成功


mybatis if test byte类型比较_sql_16


4.2. eclipse逆向工程步骤

4.2.1. 新建一个普通java项目,导入mybatis.jar包和数据库驱动包


mybatis if test byte类型比较_User_17


4.2.2. 配置生成文件


mybatis if test byte类型比较_sql_18


配置生成文件


<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE generatorConfiguration PUBLIC "-//mybatis.org//DTD MyBatis Generator Configuration 1.0//EN" "http://mybatis.org/dtd/mybatis-generator-config_1_0.dtd">
<generatorConfiguration>
  <context id="context1">
   <!-- 注释构建 -->
    <commentGenerator>
       <!-- 去掉所有的注释 -->
    	<property name="suppressAllComments" value="true"/>
    	<property name="suppressDate" value="true"/>
    </commentGenerator>
    
    <!-- 数据库四要素 -->
    <jdbcConnection connectionURL="jdbc:mysql://localhost:3306/mybatis" 
    driverClass="com.mysql.jdbc.Driver" 
    password="root" 
    userId="root" />
    <!-- 实体类 : pojo
    	targetPackage : 实体类生成后存放的包
    	targetProject : 存放的目录一般都放在 src下面
      -->
    <javaModelGenerator targetPackage="cn.zj.mybatis.pojo" targetProject="mybatis-generator/src" />
    <!-- 映射文件 -->
    <sqlMapGenerator targetPackage="cn.zj.mybatis.mapper" targetProject="mybatis-generator/src" />
    <!-- 操作接口 
    	type 生成映射的形式
    		ANNOTATEDMAPPER : 纯注解的,没有xml映射
    		XMLMAPPER : 生成的有xml映射文件
    -->
    <javaClientGenerator  targetPackage="cn.zj.mybatis.mapper" targetProject="mybatis-generator/src" type="XMLMAPPER" />
    
    <!-- 要生成对应表的配置
    	tableName : 数据库表名
    	//如果下面全部是true,mybatis直接可以使用纯面向对象开发
    	enableCountByExample : 是否生成查询总数的 Example 
    	enableDeleteByExample : 是否生成删除的 Example 
    	enableSelectByExample : 是否生成查询集合的 Example 
    	enableUpdateByExample : 是否生成修改的 Example 
     -->
    <table  tableName="user"  enableCountByExample="false" enableDeleteByExample="false" enableSelectByExample="true" enableUpdateByExample="false"></table>
    <table  tableName="employee" enableCountByExample="false" enableDeleteByExample="false" enableSelectByExample="true" enableUpdateByExample="false"></table>
    <table  tableName="department" enableCountByExample="false" enableDeleteByExample="false" enableSelectByExample="true" enableUpdateByExample="false"></table>
  </context>
</generatorConfiguration>


4.2.3. 开始逆向工程

选中 generatorConfig.xml 逆向工程配置文件,点击鼠标右键。


mybatis if test byte类型比较_sql_19


4.3. idea的逆向工程步骤(基于maven)

4.3.1. 新建一个maven项目,依赖引入mybatis、mysql-connector-java、以及mybatis-generator-maven-plugin插件


<dependencies>
        <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.38</version>
        </dependency>
    </dependencies>
 
    <build>
        <plugins>
 
            <!-- 逆向工程所需的插件 -->
            <plugin>
                <groupId>org.mybatis.generator</groupId>
                <artifactId>mybatis-generator-maven-plugin</artifactId>
                <version>1.3.2</version>
                <configuration>
                    <verbose>true</verbose>
                    <overwrite>true</overwrite>
                    <configurationFile>src/main/resources/generatorConfig.xml</configurationFile>
                </configuration>
            </plugin>
        </plugins>
    </build>


4.3.2. 编写generatorConfig.xml配置文件


<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE generatorConfiguration PUBLIC "-//mybatis.org//DTD MyBatis Generator Configuration 1.0//EN"
        "http://mybatis.org/dtd/mybatis-generator-config_1_0.dtd">
<generatorConfiguration>
    <!-- 配置驱动包的位置 -->
    <classPathEntry
            location="C:UsersJackMi.m2repositorymysqlmysql-connector-java5.1.38mysql-connector-java-5.1.38.jar" />
    <context id="context1">
        <!-- 注释构建 -->
        <commentGenerator>
            <!-- 去掉所有的注释 -->
            <property name="suppressAllComments" value="true"/>
            <property name="suppressDate" value="true"/>
        </commentGenerator>

        <!-- 数据库四要素 -->
        <jdbcConnection driverClass="com.mysql.jdbc.Driver"
                        connectionURL="jdbc:mysql://localhost:3306/mybatis"
                        userId="root"
                        password="root"/>

        <!-- 实体类 : pojo
            targetPackage : 实体类生成后存放的包
            targetProject : 存放的目录一般都放在 src下面
          -->
        <javaModelGenerator targetPackage="com.jackmi.pojo" targetProject="src/main/java"/>
        <!-- 映射文件 -->
        <sqlMapGenerator targetPackage="com.jackmi.mapper" targetProject="src/main/java"/>
        <!-- 操作接口
            type 生成映射的形式
                ANNOTATEDMAPPER : 纯注解的,没有xml映射
                XMLMAPPER : 生成的有xml映射文件
        -->
        <javaClientGenerator targetPackage="com.jackmi.mapper" targetProject="src/main/java"
                             type="XMLMAPPER"/>

        <!-- 要生成对应表的配置
            tableName : 数据库表名
            domainObjectName:生成的pojo对象的名字
            // 如果下面全部是true,mybatis直接可以使用纯面向对象开发
            enableCountByExample : 是否生成查询总数的 Example
            enableDeleteByExample : 是否生成删除的 Example
            enableSelectByExample : 是否生成查询集合的 Example
            enableUpdateByExample : 是否生成修改的 Example
         -->
        <table tableName="user" domainObjectName="User" enableCountByExample="false" enableDeleteByExample="false" 
enableSelectByExample="true" enableUpdateByExample="false"></table>
        <table tableName="employee" domainObjectName="Employee" enableCountByExample="false"
 enableDeleteByExample="false" enableSelectByExample="true" enableUpdateByExample="false"></table>
        <table tableName="department" domainObjectName="Department" enableCountByExample="false"
 enableDeleteByExample="false" enableSelectByExample="true" enableUpdateByExample="false"></table>
    </context>
</generatorConfiguration>


4.3.3. 双击mabatis-generator:generate插件


mybatis if test byte类型比较_User_20


mybatis if test byte类型比较_User_21


4.4. 逆向功能的缺点

逆向功能不能逆向多表,只能逆向单表操作,多表之间有外键对应java关联关系没办法映射,需要开发者手动编写对应代码。