1、 区分 #{} 和 ${}的不同应用场景

1)#{} 会生成预编译SQL,会正确的处理数据的类型,而${}仅仅是文本替换。

对于SQL: select * from student where xCode = ‘S123456’;

如果使用#{}

那么生成的SQL为:

select * from student where xCode = ? 传的值为’S123456’;

如果使用${}

那么生成的SQL为:select * from student where xCode = S123456

如果xCode的数据类型为varchar,那么使用${}就会报错。

2)${}一般用在order by, limit, group by等场所。

假设我们使用#{} 来指定order by字段,比如

select * from student order by #{xCode},

那么产生的SQL为

select * from student order by ?, 替换值后为

select * from student order by ‘xCode’

Mybatis对xCode加了引号导致排序失败

2、Spring环境用Mybatis-Spring的接口而不是Mybatis的原生接口

在spring 环境使用mybatis-spring的好处有:

1)我们可以使用Sping的声明式事务处理模型(@Transactional),而不用手动回归事务。

2)mybatis-spring会优雅的关闭SqlSession,而不用手动关闭

3)可以将数据库连接池交给spring管理,当程序停止的时候,spring会合适的关闭连接

3、返回Map<ID, Entity>而不是List便于查找

有时太多的表连接(join)性能太差,我们会将该SQL拆为多个SQL,然后在代码中组装起来。比如学生表和班级表,需要查询的结果为”学号,班级,姓名”,我们可以先查询“学号,班级ID,姓名”以及“班级ID,班级名次”,我们可以在查询班级表的时候返回Map<班级ID, 班级>, 然后迭代学生表的结果集,用班级ID到Map<班级ID,
班级>中查找对应的班级信息,然后用班级名称替换班级ID。

接口声明为SqlSession.selectMap(String statement, String mapKey)

4、使用Map封装查询的结果

有时我们厌倦了为每个查询写一个Entity类,这时Map开始发挥它的功效。

对于要返回“学号,班级,姓名”结果的查询,可以这样写Mapper:

1
2
3
4
5
<select id="selectStudent">
    select s.code as sNo , s.name as sName, c.name as cName
    from xStudent s, xClass c
    where s.cID = c.ID
</select>

如下声明我们的dao方法:

1
2
3
publicList<Map<String,Object>>selectStudent(Map<String,Object>parameter){
    returngetSqlSession().selectList(getStatement("selectStudent"),parameter);
}

如果要将该查询结果转为JSON字符串返回,那么我们就可以直接将List<Map<String, Object>转为JSON,逻辑层不需要任何代码。

如果返回的结果集需要按select中的字段顺序返回,那么将resultType=”java.util.HashMap” 换为resultType=”java.util.LinkedHashMap”

5、使用Map封装查询结果时注意数据的类型映射

对于如下的Mapper

1
2
3
4
<select id="selectStudent">
    select s.code as sNo , concat(s.firstName, s.lastName) as sName
    from xStudent s
</select>

Mybatis会傻傻的将sName的数据类型映射为byte[], 因为我们没有提供entity,mybatis也不知道我们想要什么类型,而sName是计算出来的值,mybatis也没有办法从数据库中获取字段的值,所以它就将其封装为byte[],解决办法很简单,加一个cast 函数

1
2
3
4
<selectid="selectStudent">
    selects.codeassNo,cast(concat(s.firstName,s.lastName)ASCHAR)assName
    fromxStudents
</select>

6、正确的配置Mybatis 的Log

1)一个应用一般会使用很多的jar,各个jar依赖的log 实现不一样,Mybatis查找Log的顺序为(SLF4J,Apache Commons Logging,Log4j 2,Log4j,JDK logging),如果classpath中有slf4j记得添加相应的桥接jar,比如slf4j-log4j。许多web
服务器的classpath 会含有Apache Commons Logging,因此如果要使用Log4j,要么使用SLF4J桥接Log4j,要么在配置中强制指定使用Log4J。

1
2
3
4
5
6
7
<configuration>
<settings>
...
<setting name="logImpl" value="LOG4J"/>
...
</settings>
</configuration>

7、警惕Mybatis的Foreach的的副作用

对于如下SQL:
假设有如下的mapper:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
<selectid=”testForeach”  parameterType=”map”  resultType=”Student”>
    Select*fromstudent
    <where>
    <iftest=”ID!=nullandID!=‘’”>
        ID=#{ID}
    </if>
    <iftest=”IDArr!=nullandIDArr.size()>0”>
        AndIDIN
            <foreachcollection="IDArr"open="("
        separator=","close=")"item="ID">
        ${ID}
            </foreach>
        </if>
</where>
</select>

当我们传入的IDArr时,最后产生的SQL为:

Select * from student where ID = ‘998’ AND ID IN ( ‘123’, ’234’,…..,’998’)

解决办法:

解决办法有

1) 将红色的ID 换成别的名称,比如“item”。

2) 这两个if 是对同一个字段判断,改为choose… when 结构

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
<select id=”testForeach”  parameterType=”map”  resultType=”Student”>
    Select * from student
    <where>
    <choose>
        <when test=”ID != null and ID != ‘’ ”>
            ID = #{ID}
        </when>
        <when test=” IDArr != null and IDArr.size()>0”>
            And ID IN
            <foreach collection="IDArr" open="("
                separator="," close=")" item="ID">
                ${ID}
            </foreach>
        </when>
</choose>
</where>

8、使用原生的SQL操作数据以提高效率

对于一次插入多条数据,将其组装成 insert into xxx values (), () ()格式一次插入多行数据往往能极大的提高性能。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
publicintexecute(Stringsql){
    intaffectedCount=0;
    Connectionconn=null;
    Statementstm=null;
    try{
        conn=getSqlSession().getConnection();
        stm=conn.createStatement();
        stm.execute(sql);
        affectedCount=stm.getUpdateCount();
    }catch(SQLExceptione){
        thrownewRuntimeException(“execute["
+ sql + "]failed”);
    }finally{
        try{
        if(stm!=null&amp;&amp;!stm.isClosed()){
        stm.close();
        }
        //conn will be released
by mybatis framework
        }catch(SQLExceptione){
        }
    }
    returnaffectedCount;
}

9、警惕MyBatis封装数据时性能损耗

对于如下的mapper

1
2
3
4
5
6
7
<select id="test" resultType="Student">
    select s.code, s.firstName,
    s.lastName, s.birthDate, s.sex, s.checkIn, s.phoneNumber,
    s.classNo
    from student
    where ...
</select>

在一个批处理程序中循环的调用了该方法250次,每次返回大概1w条记录,发现这个程序运行的很慢,用jrofiler 查看各个方法耗费的时间,居然80%的时间花在了student的setter上了,在这个过程中大概产生了250w个对象,而mybatis是利用发射封装Entity,代码大致如下:

1
2
3
4
5
6
Classc=Class.forName("cn.javacoder.testmybatis.Student");
Objecto=c.newInstance();
for(eachcolum){
Methodm=c.getMethod("get"+colum);
m.invoke(o,value);
}

解决的办法是让返回的行数和返回的字段尽量的少。

MyBatis学习总结(14)——Mybatis使用技巧总结的更多相关文章

  1. mybatis学习笔记(14)-查询缓存之中的一个级缓存

    mybatis学习笔记(14)-查询缓存之中的一个级缓存 标签: mybatis mybatis学习笔记14-查询缓存之中的一个级缓存 查询缓存 一级缓存 一级缓存工作原理 一级缓存測试 一级缓存应用 ...

  2. MyBatis学习总结(七)——Mybatis缓存(转载)

      孤傲苍狼 只为成功找方法,不为失败找借口! MyBatis学习总结(七)--Mybatis缓存 一.MyBatis缓存介绍 正如大多数持久层框架一样,MyBatis 同样提供了一级缓存和二级缓存的 ...

  3. 【转】MyBatis学习总结(七)——Mybatis缓存

    [转]MyBatis学习总结(七)——Mybatis缓存 一.MyBatis缓存介绍 正如大多数持久层框架一样,MyBatis 同样提供了一级缓存和二级缓存的支持 一级缓存: 基于PerpetualC ...

  4. 【转】MyBatis学习总结(一)——MyBatis快速入门

    [转]MyBatis学习总结(一)——MyBatis快速入门 一.Mybatis介绍 MyBatis是一个支持普通SQL查询,存储过程和高级映射的优秀持久层框架.MyBatis消除了几乎所有的JDBC ...

  5. 转:MyBatis学习总结(Mybatis总结精华文章)

    http://www.cnblogs.com/xdp-gacl/tag/MyBatis%E5%AD%A6%E4%B9%A0%E6%80%BB%E7%BB%93/ 当前标签: MyBatis学习总结   ...

  6. Mybatis学习笔记14 - 动态sql之foreach标签

    一.查询给定集合中员工id对应的所有员工信息 示例代码: 接口定义: package com.mybatis.dao; import com.mybatis.bean.Employee; import ...

  7. MyBatis 学习记录5 MyBatis的二级缓存

    主题 之前学习了一下MyBatis的一级缓存,主要涉及到BaseExecutor这个类. 现在准备学习记录下MyBatis二级缓存. 配置二级缓存与初始化发生的事情 首先二级缓存默认是不开启的,需要自 ...

  8. Mybatis学习笔记(一) —— mybatis介绍

    一.Mybatis介绍 MyBatis 本是apache的一个开源项目iBatis, 2010年这个项目由apache software foundation 迁移到了google code,并且改名 ...

  9. Mybatis学习第一天——Mybatis的安装配置以及基本CURD操作

    1.Mybatis下载 Mybatis是开源的持久层框架,能够度jdbc进行简单的封装,但其并不是完全的ORM(Object Relational Mapping,对象关系映射),无法脱离数据库进行适 ...

  10. Mybatis学习笔记(八) —— Mybatis整合spring

    一.整合思路 1.SqlSessionFactory对象应该放到spring容器中作为单例存在. 2.传统dao的开发方式中,应该从spring容器中获得sqlsession对象. 3.Mapper代 ...

随机推荐

  1. Cisco交换机IOS配置介绍

    一.模式• 用户模式(>):• 特权模式(#):• 全局配置模式(config#):• 接口配置模式(config-if#):• 线路模式(#config-line): 二.基本IOS命令结构• ...

  2. Pycharm在Ubuntu14.04中的基本使用指南

    前几天给大家分享了:如何在VMware虚拟机中安装Ubuntu14.04系统.今天给大家分享一下在Ubuntu14.04中如何简单的使用Pycharm.1.启动Pycharm,将进入Pycharm的启 ...

  3. 如何在VMware中创建虚拟机

    今天给大家分享如何在VMware中创建虚拟机,具体的教程如下.在这里小编提前下载了Ubuntu14.04桌面系统,为后面在虚拟机中安装Ubuntu14.04桌面系统做准备. 1.从官网上或者直接百度上 ...

  4. nvm安装node流程及报错解决

    第一步:下载NVM下载nvm并解压 nvm-window 下载地址:https://github.com/coreybutler/nvm-windows/releases 下载文件,然后解压得到nvm ...

  5. 学习TF:《TensorFlow技术解析与实战》PDF+代码

    TensorFlow 是谷歌公司开发的深度学习框架,也是目前深度学习的主流框架之一.<TensorFlow技术解析与实战>从深度学习的基础讲起,深入TensorFlow框架原理.模型构建. ...

  6. Java基础关于Map(字典)的方法使用

    Java基础关于Map(字典)的方法使用 java中一般用map与hashmap来创建一个key-value对象 使用前提是要导入方法包: import java.util.HashMap: impo ...

  7. 小试VS 2017 开发Python Django项目过程一

    一.新建项目python ->django web 项目 (选择带bootstrap风格与twwriter)项目名称iepiececomputing (ie计件计算)跳出窗体 -> 添加虚 ...

  8. 第四次python作业——叶耀宗

    设计题2: 1.参考“三国演义”词频统计程序,实现对红楼梦出场人物的频次统计.2.(可选)将红楼梦出场人物的频次统计结果用词云显示. import jieba excludes = {"什么 ...

  9. Centos7.6下安装Python3.7

    前言 话说不会开发的运维不是一个好的DBA,所以我要开始学习python了,写博客记录一下我的学习过程,另外别欺负我新来的,那个每天更博的技术流ken是我哥. 不说了,时间宝贵,开整. 1.首先来看一 ...

  10. Qt Installer Framework的学习(二)

    Qt Installer Framework的学习(二) Qt Installer Framework的一些操作能够使用最常见的Qt项目来表示,也就是说,书写pro文件,使用qmake执行之,除了能够 ...