一、动态SQL

相信大家在用mybatis操作数据库时时都会碰到一个问题,假如现在我们有一个关于作者的list authorList,需要根据authorList里已有的作者信息在数据库中查询相应作者的博客信息。那么最容易想到的做法就是遍历authorList,获取相应的信息查询数据库。

  1. for(int i=0;I < authorList.size();i++) {
  2. ……
  3.  
  4. //查询数据库代码
  5.  
  6. //select * from blog where author=#{author,jdbcType=VARCHAR}
  7.  
  8. }

想一想,如果假设authorList的长度为N,那么我们就需要查询N次数据库,如果用这种方法,程序的开销不仅仅是查询,还有从数据库连接池中取出连接实例、建立数据库连接、将数据库实例返还给数据库连接池,假设这三个动作加起来总共用时0.001秒。那么采取遍历的办法查询,将会多耗时0.001N秒,如果需要查询1000次,那么将多1秒钟的时间,对于程序猿来说,这是不可忍受的,因为这只是一个循环查询,还不算其它的业务代码。

那么,有没有更好的办法呢,答案是肯定,其中之一是动态SQL:

先上代码:

  1. <select id="dynamicForeachTest" resultType="com.blog.Blog" parameterType="java.util.List">
  2.  
  3. select * from blog where author in
  4.  
  5. <foreach collection="list" index="index" item="item" open="(" separator="," close=")">
  6.  
  7. #{item}
  8.  
  9. </foreach>
  10.  
  11. </select>

tem表示集合中每一个元素进行迭代时的别名,

index指定一个名字,用于表示在迭代过程中,每次迭代到的位置,

open表示该语句以什么开始,

separator表示在每次进行迭代之间以什么符号作为分隔符,

close表示以什么结束这样返回值就可以用List<Bolg>接受.

但是动态SQL中的foreach语句用的最多的实在insert语句中,并且通常在in子句中使用。

二、高级映射

在使用mybatis的时候,一般是使用resultType = com.blog.author 实体类来接受查询结果

或者是使用resultType = java.util.map将数据库列名作为key,记录值作为value返回。

但是这次需要使用resultMap,它可以允许自由组合返回值的形式,用以处理更复杂的查询。

还是先上代码:

SQL:

  1. <select id="getBlogs" resultMap=" blogs " parameterType="map">
  2. Select a.authorID,
  3. a.uthorName,
  4. b.blogID,
  5. b.blogName
  6. from author a left join blog b on a. authorID=b. authorID where a. authorID = #{authorID,jdbcType=INTEGER}
  7. </select>

mybatis配置:

  1. <resultMap id="blogs" type="com.bloh.Blog">
  2.  
  3. <id property="authorID" column=" authorID">
  4.  
  5. <result property="authorName" column=" authorName">
  6.  
  7. <collection property="postsList" ofType="com.bolg.Post">
  8.  
  9. <id property="blogID" column=" blogID"/>
  10.  
  11. <result property="blogName" column="blogName"/>
  12.  
  13. </collection>
  14.  
  15. </resultMap>

Blog实体类

  1. Public class Bolg {
  2.  
  3. private Integer authorID;
  4.  
  5. private String authorName;
  6.  
  7. private List<Post> postsList;
  8.  
  9. //setter getter
  10.  
  11. }

Post实体类

  1. Public class Post {
  2.  
  3. private Integer blogID;
  4.  
  5. private String blogName;
  6.  
  7. //setter getter
  8.  
  9. }

这样就可以用一个实体接受一个复杂查询了。

下面再介绍下各个属性的作用:

其它和普通mybatis查询的属性和配置就不细说了,

resultMap用来代替resultType,表示查询结果返回的格式

resultMap中的id主要有两个作用:

  1. 类似索引,提高查询性能
  2. 区分不同结果

所以id最好不要省略,如果没有主键,用能唯一区分记录的字段代替

result即实体类中定义的变量名,column是数据库的列名

collection 就是列表、map等集合

postsList就是在Blog实体类中定义的list变量名

ofType就是对象列表中对象的实体类。

三、获得自增ID:

如果有如下情况,在插入数据库记录后,想得到插入记录的主键,用以后面的业务代码

那么mybatis针对这种情况也提供了相应的支持(不支持批量插入):

MySQL是原声自增ID;假设自增主键的字段名就为ID

  1. <insert id="insert" useGeneratedKeys="true" keyProperty="id" parameterType="User">
  2.  
  3. insert into <include refid="TABLE_NAME" /> ( NAME, AGE )
  4.  
  5. values ( #{name}, #{age} )
  6.  
  7. </insert>

比普通的插入就多了两个属性 useGeneratedKeys="true" 表示开启返回自增ID

keyProperty="id" 表示返回主键的名字。

那么在业务代码中就可以用下列语句接收:

假设实体类为User

  1. User userNew = userMapper.insert(user);
  2.  
  3. userNew.getID //即为插入后的自增ID

其实,mysql的自增主键可以用select LAST_INSERT_ID();来得到,

所以,还有一种写法:

  1. <insert id="insert" parameterType="User">
  2.  
  3. <selectKey resultType="int" order="AFTER" keyProperty="id">
  4. SELECT LAST_INSERT_ID() AS id
  5. </selectKey>
  6.  
  7. insert into name,age
  8.  
  9. values ( #{name}, #{age} )
  10.  
  11. </insert>

和mysql的获取主键方式刚好相反,mysql是insert执行后由表分配自增长的值,而oracle是获取到自增长的值后再进行插入记录操作,在执行insert sql前必须指定一个主键值给要插入的记录所以要要在"BEFORE"的时候拿到自增的序列,然后用selectKey的方式注入到入参映射中即可。假设自增长还是id

  1. <insert id=" insert " useGeneratedKeys="true" keyProperty="id" parameterType="xxxx" >
  2.  
  3. <selectKey resultType="int" order="BEFORE" keyProperty="id">
  4.  
  5. SELECT SEQ_TABLE.NEXTVAL FROM dual
  6.  
  7. </selectKey>
  8.  
  9. INSERT INTO id,name,age
  10.  
  11. VALUES
  12.  
  13. (#{id} #{name}, #{age} )
  14.  
  15. </insert>

这里的id就是selectKey获得的自增id。

接收方式和mysql一样,在获取自增主键时,最好使用实体接收。

Mybatis高级映射、动态SQL及获得自增主键的更多相关文章

  1. (转)Mybatis高级映射、动态SQL及获得自增主键

    原文:http://www.cnblogs.com/edwinchen/p/4105278.html?utm_source=tuicool&utm_medium=referral 一.动态SQ ...

  2. MyBatis 3.2.6插入时候获取自增主键方法

    MyBatis 3.2.6插入时候获取自增主键方法有二 以MySQL5.5为例: 方法1: <insert id="insert" parameterType="P ...

  3. 关于mybatis用mysql时,插入返回自增主键的问题

    公司决定新项目用mybatis,虽然这个以前学过但是一直没用过都忘得差不多了,而且项目比较紧,也没时间去系统点的学一学,只好很粗略的百度达到能用的程度就行了. 其中涉及到插入实体要求返回主键id的问题 ...

  4. mybatis执行insert后马上能获取自增主键的语句写法

    <!--keyColumn keyProperty useGeneratedKeys 用于在插入数据后,能直接使用user.getId()获取主键--> <insert id=&qu ...

  5. Mybatis批量插入返回自增主键(转)

    我们都知道Mybatis在插入单条数据的时候有两种方式返回自增主键: 1.对于支持生成自增主键的数据库:useGenerateKeys和keyProperty. 2.不支持生成自增主键的数据库:< ...

  6. 【Mybatis高级映射】一对一映射、一对多映射、多对多映射

    前言 当我们学习heribnate的时候,也就是SSH框架的网上商城的时候,我们就学习过它对应的高级映射,一对一映射,一对多映射,多对多映射.对于SSM的Mybatis来说,肯定也是差不多的.既然开了 ...

  7. mybatis高级映射(一对一,一对多)

    mybatis高级映射 一对一关联映射 需求:查询订单信息,关联查询用户信息(一个订单对应一个用户) (1)通过resultType实现 sql语句: select orders.* , USER.u ...

  8. 【mybatis深度历险系列】mybatis中的动态sql

    最近一直做项目,博文很长时间没有更新了,今天抽空,学习了一下mybatis,并且总结一下.在前面的博文中,小编主要简单的介绍了mybatis中的输入和输出映射,并且通过demo简单的介绍了输入映射和输 ...

  9. mybatis 高级映射和spring整合之逆向工程(7)

    mybatis 高级映射和spring整合之逆向工程(7) 4.0 逆向工程 4.1 mybatis需要程序员自己编写sql语句,mybatis官方提供逆向工程,可以针对单表自动生成mybatis执行 ...

随机推荐

  1. smokeping报错Can't locate RRDs.pm in @INC (@INC contains

    安装完smokeping,执行debug语句: ./bin/smokeping --debug-daemon ,提示如下错误: Can't locate RRDs.pm in @INC (@INC c ...

  2. php resizeimage 部分jpg文件 生成缩略图失败

    今天遇到GD的resizeimage 函数处理jpg后缀文件的缩略图的时候 提示该图片不是合法的jpg图片并报错 <b>Warning</b>: imagecreatefrom ...

  3. php批量上传图片并把图片名放入数据库

    前几天工作中要做这样一个功能,有八百多个系统 生成的会员:给这八百多个系统会员上传图片:然后把图片名放入数据库. 第一步: 第一步肯定是首先把图片上传到对应的图片目录下,直接用框架中已经有的上传类: ...

  4. 如何使用sublime编辑器运行python程序

    现在越发喜欢sublime编辑器了,不仅界面友好美观.文艺,可扩展性还特别强. sublime本身是不具备运行python程序的能力的,需要做些设置才可以.以下是安装好sublime后设置的步骤: 点 ...

  5. Array and its point.

    a is the array name. &a is the ponit of 2-D array which contains a[5]. the type of &a should ...

  6. 模糊系统架构和简单实现--AForge.NET框架的使用(四)

    原文:模糊系统架构和简单实现--AForge.NET框架的使用(四) 先说一下,为什么题目是简单实现,因为我实在没有弄出好的例子. 我原来用AForge.net做的项目中的模糊系统融入了神经网络和向量 ...

  7. MyBatis查询两个字段,返回Map,一个字段作为key,一个字段作为value的实现

    1. 问题描述 在使用MyBatis,我们经常会遇到这种情况:SELECT两个字段,需要返回一个Map,其中第一个字段作为key,第二个字段作为value.MyBatis的MapKey虽然很实用,但并 ...

  8. js中()()问题

    var aa=function(){}(); var bb=(function(){})(); 今天被问到这个问题,这段js有撒区别. 总结一下,两个函数都是立即执行的意思.但是不同之处是执行的顺序, ...

  9. COJ 0359 xjr考考你数据结构(根号2)线段树区间增加

    xjr考考你数据结构(根号2) 难度级别:C: 运行时间限制:3000ms: 运行空间限制:51200KB: 代码长度限制:2000000B 试题描述 请你编写一个数据结构,完成以下功能: 1)求出第 ...

  10. BZOJ1690: [Usaco2007 Dec]奶牛的旅行

    1690: [Usaco2007 Dec]奶牛的旅行 Time Limit: 5 Sec  Memory Limit: 64 MBSubmit: 552  Solved: 286[Submit][St ...