使Mybatis开发变得更加轻松的增强工具 — Ourbatis

一、Mybatis的不足之处

Mybatis是一款优秀的及其灵活的持久层框架,通过XML配置并映射到Mapper接口为Service层提供基础数据操作入口。

这么优秀的框架竟然还有不足之处?

俗话说人无完人,因为Mybatis实在是太灵活了,灵活到每个Mapper接口都需要定制对应的XML,所以就会引发一些问题。

问题一:配置文件繁多

假如一个系统中DB中涉及100张表,我们就需要写100个Mapper接口,还没完,最可怕的是,我们要为这100个Mapper接口定制与之对应的100套XML。而每个Mapper都必不可少的需要增删改查功能,我们就要写100遍增删改查,作为高贵的Java开发工程师,这是不能容忍的,于是Mybatis Generator诞生了,然而又会引发另一个问题!

问题二:维护困难

我们使用Mybatis Generator解决了问题一,再多的文件生成就是了,简单粗暴,貌似解决了所有的问题,Mybatis完美了!

不要高兴的太早,在系统刚刚建立起来时,我们使用Mybatis Generator生成了一堆XML,在开发过程中,产品忽然提了一个新的需求,项目经理根据这个需求在某张表中增加或变动了一个字段,这时,我猜你的操作是这样:

  • 1、找到对应表的XML
  • 2、将该XML中自定义的一段标签复制出来,保存在本地
  • 3、使用Mybatis Generator重新生成该表的XML
  • 4、将之覆盖当前的XML
  • 5、将自定义的一段标签再粘贴进新的XML中

在这个过程中,如果我们在第2步时漏复制了一段标签,等整个操作完成之后,又别是一番滋味在心头~

问题三:编写XML困难

假如肝不错,问题二也是小CASE,那么问题又来了,我们如何在繁长的XML中去编写和修改我们的XML呢。

当我们打开要编辑的XML,映入眼帘的就是1000多行的XML,其中900行都是通用的增删改查操作,要新增一个标签,我们需要拉至文件底部编写新的数据操作,要更新一个标签,我们需要通过Ctrl + F寻找目标标签再进行修改。

如何避免这些问题呢?

如何让Mybatis增强通用性又不失灵活呢?

二、使用Ourbatis辅助Mybatis

Ourbatis是一款Mybatis开发增强工具,小巧简洁,项目地址:

特性:

  • 1、简洁方便,可以让Mybatis无XML化开发。
  • 2、优雅解耦,通用和自定义的SQL标签完全隔离,让维护更加轻松。
  • 3、无侵入性,Mybatis和Ourbatis可同时使用,配置简洁。
  • 4、灵活可控,通用模板可自定义及扩展。
  • 5、部署快捷,只需要一个依赖,两个配置,即可直接运行。
  • 6、多数据源,在多数据源环境下也可以照常使用。

关于Ourbatis使用的一个小Demo

环境:

  • Spring Boot 2.0.5.RELEASE
  • Ourbatis 1.0.5
  • JAVA 8
  • Mysql

Spring Boot 2.0.5.RELEASE版本为例,在可以正常使用Mybatis的项目中,pom.xml添加如下依赖:

   <dependency>
           <groupId>com.smallnico</groupId>
           <artifactId>ourbatis-spring-boot-starter</artifactId>
           <version>1.0.5</version>
   </dependency>

在配置文件中增加一下配置:

ourbatis.domain-locations=实体类所在包名

接下来,Mapper接口只需要继承SimpleMapper即可:

import org.nico.ourbatis.domain.User;
public interface UserMapper extends SimpleMapper<User, Integer>{
}

至此,一个使用Ourbatis的简单应用已经部署起来了,之后,你就可以使用一些Ourbatis默认的通用操作方法:

    public T selectById(K key);
    
    public T selectEntity(T condition);
    
    public List<T> selectList(T condition);
    
    public long selectCount(Object condition);
    
    public List<T> selectPage(Page<Object> page);
    
    default PageResult<T> selectPageResult(Page<Object> page){
        long total = selectCount(page.getEntity());
        List<T> results = null;
        if(total > 0) {
            results = selectPage(page);
        }
        return new PageResult<>(total, results);
    }
    
    public K selectId(T condition);
    
    public List<K> selectIds(T condition);
    
    public int insert(T entity);
    
    public int insertSelective(T entity);
    
    public int insertBatch(List<T> list);
    
    public int update(T entity);
    
    public int updateSelective(T entity);
    
    public int updateBatch(List<T> list);
    
    public int delete(T condition);
    
    public int deleteById(K key);
    
    public int deleteBatch(List<K> list);

Mapper自定义方法

在很多场景中,我们使用以上的自带的通用方法远远不能满足我们的需求,我们往往需要额外扩展新的Mapper方法、XML标签,我们使用了Ourbatis之后该如何实现呢?

首先看一下我们的需求,在上述Demo中,我们在UserMapper中增加一个方法selectNameById

import org.nico.ourbatis.domain.User;
public interface UserMapper extends SimpleMapper<User, Integer>{
    public String selectNameById(Integer userId);
}

和Mybatis一样,需要在resources资源目录下新建一个文件夹ourbatis-mappers,然后在其中新建一个XML文件,命名规则为:

DomainClassSimpleName + Mapper.xml

其中DomainClassSimpleName就是我们实体类的类名,这里是为User,那么新建的XML名为UserMapper.xml

src/main/resources
 - ourbatis-mappers
   - UserMapper.xml

之后,打开UserMapper.xml,开始编写Mapper中selectNameById方法对应的标签:

<select id="selectNameById" resultType="java.lang.String">
    select name from user where id = #{userId}
</select>

注意,整个文件中只需要写标签就行了,其他的什么都不需要,这是为什么呢?深入之后你就会明白,这里先不多说!

接下来,就没有接下来了,可以直接使用selectNameById方法了。

深入了解Ourbatis

ourbatis 流程图

当服务启动的时候,Ourbatis首先会扫描ourbatis.domain-locations配置包下的所有实体类,将之映射为与之对应的表结构数据:

ourbatis Mapping

然后通过ourbatis.xml的渲染,生成一个又一个的XML文件,最后将之重新Build到Mybatis容器中!

整个过程分为两个核心点:

  • 1、映射实体类为元数据
  • 2、使用ourbatis.xml渲染元数据为XML文件

我会一一介绍之~

映射实体类为元数据

在映射时,我们要根据自己数据库字段命名的风格去调整映射规则,就需要在第1个核心点中去做处理,Ourbatis使用包装器来完成:

public interface Wrapper<T> {
    public String wrapping(T value);
}

对于需要映射的字段,如表名表字段名,它们都将会经过一个包装器链条的处理之后再投入到ourbatis.xml中做渲染,这样就使得我们可以自定义包装器出更换映射的字段格式,具体方式可以参考官方Wiki:Wrapper包装器

使用ourbatis.xml渲染元数据为XML文件

而在于第2个核心点中,Ourbatis通过自定义标签做模板渲染,我们可以先看一下官方默认的ourbatis.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="@{mapperClassName}">
    <resultMap id="BaseResultMap" type="@{domainClassName}">
        <ourbatis:foreach list="primaryColumns" var="elem">
            <id column="@{elem.jdbcName}" property="@{elem.javaName}" />
        </ourbatis:foreach>
        <ourbatis:foreach list="normalColumns" var="elem">
            <result column="@{elem.jdbcName}" property="@{elem.javaName}" />
        </ourbatis:foreach>
    </resultMap>

    <sql id="Base_Column_List">
        <ourbatis:foreach list="allColumns" var="elem"
            split=",">
            `@{elem.jdbcName}`
        </ourbatis:foreach>
    </sql>

    <select id="selectById" parameterType="java.lang.Object"
        resultMap="BaseResultMap">
        select
        <include refid="Base_Column_List" />
        from @{tableName}
        where 1 = 1
        <ourbatis:foreach list="primaryColumns" var="elem">
            and `@{elem.jdbcName}` = #{@{elem.javaName}}
        </ourbatis:foreach>
    </select>

    <select id="selectEntity" parameterType="@{domainClassName}"
        resultMap="BaseResultMap">
        select
        <include refid="Base_Column_List" />
        from @{tableName}
        where 1 = 1
        <ourbatis:foreach list="allColumns" var="elem">
            <if test="@{elem.javaName} != null">
                and `@{elem.jdbcName}` = #{@{elem.javaName}}
            </if>
        </ourbatis:foreach>
        limit 1
    </select>

    <select id="selectCount" parameterType="@{domainClassName}"
        resultType="long">
        select count(0)
        from @{tableName}
        where 1 = 1
        <ourbatis:foreach list="allColumns" var="elem">
            <if test="@{elem.javaName} != null">
                and `@{elem.jdbcName}` = #{@{elem.javaName}}
            </if>
        </ourbatis:foreach>
        limit 1
    </select>

    <select id="selectPage"
        parameterType="org.nico.ourbatis.entity.Page"
        resultMap="BaseResultMap">
        select
        <include refid="Base_Column_List" />
        from @{tableName}
        where 1 = 1
        <if test="entity != null">
            <ourbatis:foreach list="allColumns" var="elem">
                <if test="entity.@{elem.javaName} != null">
                    and `@{elem.jdbcName}` = #{entity.@{elem.javaName}}
                </if>
            </ourbatis:foreach>
        </if>
        <if test="orderBy != null">
            order by ${orderBy}
        </if>
        <if test="start != null and end != null">
            limit ${start},${end}
        </if>
    </select>

    <select id="selectList" parameterType="@{domainClassName}"
        resultMap="BaseResultMap">
        select
        <include refid="Base_Column_List" />
        from @{tableName}
        where 1 = 1
        <ourbatis:foreach list="allColumns" var="elem">
            <if test="@{elem.javaName} != null">
                and `@{elem.jdbcName}` = #{@{elem.javaName}}
            </if>
        </ourbatis:foreach>
    </select>

    <select id="selectId" parameterType="@{domainClassName}"
        resultType="java.lang.Object">
        select
        <ourbatis:foreach list="primaryColumns" var="elem"
            split=",">
            `@{elem.jdbcName}`
        </ourbatis:foreach>
        from @{tableName}
        where 1 = 1
        <ourbatis:foreach list="allColumns" var="elem">
            <if test="@{elem.javaName} != null">
                and `@{elem.jdbcName}` = #{@{elem.javaName}}
            </if>
        </ourbatis:foreach>
        limit 1
    </select>

    <select id="selectIds" parameterType="@{domainClassName}"
        resultType="java.lang.Object">
        select
        <ourbatis:foreach list="primaryColumns" var="elem"
            split=",">
            `@{elem.jdbcName}`
        </ourbatis:foreach>
        from @{tableName}
        where 1 = 1
        <ourbatis:foreach list="normalColumns" var="elem">
            <if test="@{elem.javaName} != null">
                and `@{elem.jdbcName}` = #{@{elem.javaName}}
            </if>
        </ourbatis:foreach>
    </select>

    <delete id="deleteById" parameterType="java.lang.Object">
        delete
        from @{tableName}
        where 1=1
        <ourbatis:foreach list="primaryColumns" var="elem">
            and `@{elem.jdbcName}` = #{@{elem.javaName}}
        </ourbatis:foreach>
    </delete>

    <insert id="insert" keyProperty="@{primaryColumns.0.jdbcName}"
        useGeneratedKeys="true" parameterType="@{domainClassName}">
        insert into @{tableName}
        (
        <include refid="Base_Column_List" />
        )
        values (
        <ourbatis:foreach list="allColumns" var="elem"
            split=",">
            #{@{elem.javaName}}
        </ourbatis:foreach>
        )
    </insert>

    <insert id="insertSelective"
        keyProperty="@{primaryColumns.0.jdbcName}" useGeneratedKeys="true"
        parameterType="@{domainClassName}">
        insert into @{tableName}
        (
        <ourbatis:foreach list="primaryColumns" var="elem"
            split=",">
            `@{elem.jdbcName}`
        </ourbatis:foreach>
        <ourbatis:foreach list="normalColumns" var="elem">
            <if test="@{elem.javaName} != null">
                ,`@{elem.jdbcName}`
            </if>
        </ourbatis:foreach>
        )
        values (
        <ourbatis:foreach list="primaryColumns" var="elem">
            #{@{elem.javaName}}
        </ourbatis:foreach>
        <ourbatis:foreach list="normalColumns" var="elem">
            <if test="@{elem.javaName} != null">
                , #{@{elem.javaName}}
            </if>
        </ourbatis:foreach>
        )
    </insert>

    <insert id="insertBatch"
        keyProperty="@{primaryColumns.0.jdbcName}" useGeneratedKeys="true"
        parameterType="java.util.List">
        insert into @{tableName}
        (
        <include refid="Base_Column_List" />
        )
        values
        <foreach collection="list" index="index" item="item"
            separator=",">
            (
            <ourbatis:foreach list="allColumns" var="elem"
                split=",">
                #{item.@{elem.javaName}}
            </ourbatis:foreach>
            )
        </foreach>
    </insert>

    <update id="update" parameterType="@{domainClassName}">
        update @{tableName}
        <set>
            <ourbatis:foreach list="normalColumns" var="elem"
                split=",">
                `@{elem.jdbcName}` = #{@{elem.javaName}}
            </ourbatis:foreach>
        </set>
        where 1=1
        <ourbatis:foreach list="primaryColumns" var="elem">
            and `@{elem.jdbcName}` = #{@{elem.javaName}}
        </ourbatis:foreach>
    </update>

    <update id="updateSelective" parameterType="@{domainClassName}">
        update @{tableName}
        <set>
            <ourbatis:foreach list="primaryColumns" var="elem"
                split=",">
                `@{elem.jdbcName}` = #{@{elem.javaName}}
            </ourbatis:foreach>
            <ourbatis:foreach list="normalColumns" var="elem">
                <if test="@{elem.javaName} != null">
                    ,`@{elem.jdbcName}` = #{@{elem.javaName}}
                </if>
            </ourbatis:foreach>
        </set>
        where 1=1
        <ourbatis:foreach list="primaryColumns" var="elem">
            and `@{elem.jdbcName}` = #{@{elem.javaName}}
        </ourbatis:foreach>
    </update>

    <update id="updateBatch" parameterType="java.util.List">
        <foreach collection="list" index="index" item="item"
            separator=";">
            update @{tableName}
            <set>
                <ourbatis:foreach list="normalColumns" var="elem"
                    split=",">
                    `@{elem.jdbcName}` = #{item.@{elem.javaName}}
                </ourbatis:foreach>
            </set>
            where 1=1
            <ourbatis:foreach list="primaryColumns" var="elem">
                and `@{elem.jdbcName}` = #{item.@{elem.javaName}}
            </ourbatis:foreach>
        </foreach>
    </update>

    <delete id="deleteBatch" parameterType="java.util.List">
        delete from @{tableName} where @{primaryColumns.0.jdbcName} in
        <foreach close=")" collection="list" index="index" item="item"
            open="(" separator=",">
            #{item}
        </foreach>
    </delete>

    <delete id="delete" parameterType="@{domainClassName}">
        delete from @{tableName} where 1 = 1
        <ourbatis:foreach list="allColumns" var="elem">
            <if test="@{elem.javaName} != null">
                and `@{elem.jdbcName}` = #{@{elem.javaName}}
            </if>
        </ourbatis:foreach>
    </delete>

    <ourbatis:ref path="classpath:ourbatis-mappers/@{domainSimpleClassName}Mapper.xml" />
</mapper>

可以看出来,ourbatis.xml内容类似于原生的Mybatis的XML,不同的是,有两个陌生的标签:

  • ourbatis:foreach 对元数据中的列表进行循环渲染
  • ourbatis:ref 引入外界文件内容

这是Ourbatis中独有的标签,Ourbatis也提供着对应的入口让我们去自定义标签:

Class: org.nico.ourbatis.Ourbatis
Field: 
public static final Map<String, AssistAdapter> ASSIST_ADAPTERS = new HashMap<String, AssistAdapter>(){
        private static final long serialVersionUID = 1L;
        {
            put("ourbatis:foreach", new ForeachAdapter());
            put("ourbatis:ref", new RefAdapter());
        }
    };

我们可以修改org.nico.ourbatis.Ourbatis类中的静态参数ASSIST_ADAPTERS去删除、更新和添加自定义标签,需要实现一个标签适配器,我们可以看一下最简单的RefAdapter适配器的实现:

public class RefAdapter extends AssistAdapter{
    @Override
    public String adapter(Map<String, Object> datas, NoelRender render, Document document) {
        String path = render.rending(datas, document.getParameter("path"), "domainSimpleClassName");
        String result =  StreamUtils.convertToString(path.replaceAll("classpath:", ""));
        return result == null ? "" : result.trim();
    }
}

Ourbatis中只定义了上述两个自定义标签已足够满足需求,通过foreach标签,将元数据中的集合遍历渲染,通过ref标签引入外界资源,也就是我们之前所说的对Mapper接口中方法的扩展!

<ourbatis:ref path="classpath:ourbatis-mappers/@{domainSimpleClassName}Mapper.xml" />

其中的path就是当前项目classpath路径的相对路径,而@{domainSimpleClassName}就代表着实体类的类名,更多的系统参数可以参考Wiki:元数据映射

通过这种模板渲染的机制,Ourbatis是相当灵活的,我们不仅可以通过引入外部文件进行扩展,当我们需要添加或修改通用方法时,我们可以可以自定义ourbatis.xml的内容,如何做到呢?复制一份将之放在资源目录下就可以了!

看到这里,相信大家已经知道Ourbatis的基本原理已经使用方式,我就再次不多说了,更多细节可以去官方Wiki中阅读:Ourbtis Wiki

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,463评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,868评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,213评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,666评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,759评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,725评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,716评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,484评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,928评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,233评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,393评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,073评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,718评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,308评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,538评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,338评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,260评论 2 352

推荐阅读更多精彩内容

  • 一、Mybatis的不足之处 Mybatis是一款优秀的及其灵活的持久层框架,通过XML配置并映射到Mapper接...
    Nico_880f阅读 693评论 0 0
  • 1 Mybatis入门 1.1 单独使用jdbc编程问题总结 1.1.1 jdbc程序 上边使...
    哇哈哈E阅读 3,303评论 0 38
  • 1. 简介 1.1 什么是 MyBatis ? MyBatis 是支持定制化 SQL、存储过程以及高级映射的优秀的...
    笨鸟慢飞阅读 5,502评论 0 4
  • 廿载犹劳好护持,气粗语大旧吟诗;而今律细才偏退,可许情怀似昔时。 【闫言:钱锺书此言自己年轻时恃才傲物,多亏杨绛理...
    闫言阅读 2,426评论 0 2
  • 临在让我有很多力量!前天下午3点接到管委会通知让我第二天早上9点多给200人上一个新出台的财政补贴课程,说的是只要...
    赵爱丽阅读 123评论 0 1