ResultMap

在实际的开发中,数据库不总是我们希望看到的样子。比如我们希望User的主键是id但是数据库偏偏喜欢叫它u_id,这样一来原先的resultType似乎就失效了,不带这么玩的,整个人都不好了。

于是mybatis给出了他的方案——resultMap。把我们从复杂的命名问题中解救出来~~~

在上一篇中已经用mybatis generator生成好了一个BlogMapper.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="pro.app.inter.BlogMapper" >
<resultMap id="BaseResultMap" type="pro.app.model.Blog" >
<id column="b_id" property="bId" jdbcType="INTEGER" />
<result column="b_title" property="bTitle" jdbcType="VARCHAR" />
<result column="b_content" property="bContent" jdbcType="VARCHAR" />
<result column="user_id" property="userId" jdbcType="INTEGER" />
</resultMap> <sql id="Base_Column_List" >
b_id, b_title, b_content, user_id
</sql>
<select id="selectByPrimaryKey" resultMap="BaseResultMap" parameterType="java.lang.Integer" >
select
<include refid="Base_Column_List" />
from blog
where b_id = #{bId,jdbcType=INTEGER}
</select>
<delete id="deleteByPrimaryKey" parameterType="java.lang.Integer" >
delete from blog
where b_id = #{bId,jdbcType=INTEGER}
</delete>
<insert id="insert" parameterType="pro.app.model.Blog" >
<selectKey resultType="java.lang.Integer" keyProperty="bId" order="AFTER" >
SELECT LAST_INSERT_ID()
</selectKey>
insert into blog (b_title, b_content, user_id
)
values (#{bTitle,jdbcType=VARCHAR}, #{bContent,jdbcType=VARCHAR}, #{userId,jdbcType=INTEGER}
)
</insert>
<insert id="insertSelective" parameterType="pro.app.model.Blog" >
<selectKey resultType="java.lang.Integer" keyProperty="bId" order="AFTER" >
SELECT LAST_INSERT_ID()
</selectKey>
insert into blog
<trim prefix="(" suffix=")" suffixOverrides="," >
<if test="bTitle != null" >
b_title,
</if>
<if test="bContent != null" >
b_content,
</if>
<if test="userId != null" >
user_id,
</if>
</trim>
<trim prefix="values (" suffix=")" suffixOverrides="," >
<if test="bTitle != null" >
#{bTitle,jdbcType=VARCHAR},
</if>
<if test="bContent != null" >
#{bContent,jdbcType=VARCHAR},
</if>
<if test="userId != null" >
#{userId,jdbcType=INTEGER},
</if>
</trim>
</insert>
<update id="updateByPrimaryKeySelective" parameterType="pro.app.model.Blog" >
update blog
<set >
<if test="bTitle != null" >
b_title = #{bTitle,jdbcType=VARCHAR},
</if>
<if test="bContent != null" >
b_content = #{bContent,jdbcType=VARCHAR},
</if>
<if test="userId != null" >
user_id = #{userId,jdbcType=INTEGER},
</if>
</set>
where b_id = #{bId,jdbcType=INTEGER}
</update>
<update id="updateByPrimaryKey" parameterType="pro.app.model.Blog" >
update blog
set b_title = #{bTitle,jdbcType=VARCHAR},
b_content = #{bContent,jdbcType=VARCHAR},
user_id = #{userId,jdbcType=INTEGER}
where b_id = #{bId,jdbcType=INTEGER}
</update>
</mapper>

在文件的开头位置,就能够发现一个名为BaseResultMap的resultMap标签。在这个标签当中我们可以发现几个子标签。这几个子标签对通过column和property属性将数据库字段和model对应类型关联起来。而<id/>标签代表这个model类的属性对应的数据库字段为这张表的主键。定义完成这个BaseResultMap之后,我们就可以在后面的标签对中使用它作为返回的结果使用。 这里可以注意到select的属性resultMap="BaseResultMap" 。返回的数据通过resultMap被封装成了相应的对象,如果返回的数据是多条,mybatis也会自动将结果集转换为List集合。

这里还可以关注下resultMap下的sql标签对,在这个标签对中可以写入数据库的字段名称。在CRUD的xml文件中使用<include/>标签引用这它,在数据库需要修改的时候,我们就不用去修改这个配置文件下每一个sql语句。

数据关联

有人会说。这样resultMap和resultType有什么区别!还要多写这么一堆配置,这不是吃饱了撑着么!!!真的是这个样子?

我们之前设计了两张表,一个用户表和一个博客列表,一个用户可以有多篇博客吧,一篇博客只有一个作者。现在就让resultMap帮助我们把数据关联起来~

先给表Blog添加几条数据

 INSERT INTO `blog`.`blog` (`b_id`, `b_title`, `b_content`, `user_id`) VALUES ('', '', 'mybatis001', '');
INSERT INTO `blog`.`blog` (`b_id`, `b_title`, `b_content`, `user_id`) VALUES ('', '', 'mybatis002', '');
INSERT INTO `blog`.`blog` (`b_id`, `b_title`, `b_content`, `user_id`) VALUES ('', '', 'mybatis003', '');
INSERT INTO `blog`.`blog` (`b_id`, `b_title`, `b_content`, `user_id`) VALUES ('', '', 'mybatis004', '');
INSERT INTO `blog`.`blog` (`b_id`, `b_title`, `b_content`, `user_id`) VALUES ('', '', 'mybatis005', '');

在此之前我们需要再定义一个BlogVo类,并继承Blog类,然后添加一个User类型属性user。

 package pro.app.model;

 public class BlogVo extends Blog{
private User user;
public User getUser() {
return user;
}
public void setUser(User user) {
this.user = user;
}
}

接着在BlogMapper.xml中添加一个resultMap。

 <resultMap id="BaseResultMapWithUser" type="pro.app.model.BlogVo" >
<id column="b_id" property="bId" jdbcType="INTEGER" />
<result column="b_title" property="bTitle" jdbcType="VARCHAR" />
<result column="b_content" property="bContent" jdbcType="VARCHAR" />
<association property="user" javaType="pro.app.model.User">
<id column="user_id" property="id" jdbcType="INTEGER" />
<result column="name" property="name" jdbcType="VARCHAR" />
<result column="age" property="age" jdbcType="INTEGER" />
</association>
</resultMap>

在resultMap中通过子标签association,我们将外键与对应的model类型对应起来。用association中的property属性对应java属性的用户。association下的子标签id对应blog表中的外键,这也是数据关联的关键!接着再给这个文件添加一个select标签。

  <select id="selectByPrimaryKeyWithUser" resultMap="BaseResultMapWithUser" parameterType="java.lang.Integer" >
select *
from blog b
join user u
on b.user_id=u.id
where b.b_id= #{id,jdbcType=INTEGER}
</select>

ok,让我们在BlogMapper.java里添加一个selectByPrimaryKeyWithUser()方法。

 package pro.app.inter;

 import org.apache.ibatis.annotations.Param;

 import pro.app.model.Blog;
import pro.app.model.BlogVo; public interface BlogMapper {
int deleteByPrimaryKey(Integer bId); int insert(Blog record); int insertSelective(Blog record); Blog selectByPrimaryKey(Integer bId); int updateByPrimaryKeySelective(Blog record); int updateByPrimaryKey(Blog record); BlogVo selectByPrimaryKeyWithUser(@Param("id")Integer id); }

建立一个测试类来看看效果

 package pro.test;

 import java.io.Reader;

 import org.apache.ibatis.io.Resources;
import org.apache.ibatis.session.SqlSession;
import org.apache.ibatis.session.SqlSessionFactory;
import org.apache.ibatis.session.SqlSessionFactoryBuilder; import pro.app.inter.BlogMapper;
import pro.app.model.Blog;
import pro.app.model.BlogVo; public class BlogVoTest {
private static SqlSessionFactory sqlSessionFactory;
private static Reader reader;
static{
try{
reader= Resources.getResourceAsReader("Configuration.xml");
sqlSessionFactory = new SqlSessionFactoryBuilder().build(reader);
}catch(Exception e){
e.printStackTrace();
}
}
public static SqlSessionFactory getSession(){
return sqlSessionFactory;
}
public static void main(String[] args) {
SqlSession session = sqlSessionFactory.openSession();
try {
BlogMapper blogs = session.getMapper(BlogMapper.class);
BlogVo bv=blogs.selectByPrimaryKeyWithUser(1);
System.out.println(bv.getUser().getName());
} finally {
session.close();
}
}
}

控制台输出了:

Mybatis

ok!mybatis帮我们把博客的作者选出来了。现在我们来通过作者选出他所有的文章。

同样我们在UserMapper.xml里添加一个resultMap

   <resultMap id="BaseResultMapWithBlogs" type="pro.app.model.UserVo" >
<id column="id" property="id" jdbcType="INTEGER" />
<result column="name" property="name" jdbcType="VARCHAR" />
<result column="age" property="age" jdbcType="INTEGER" />
<collection property="blogs" ofType="pro.app.model.Blog" column="b_id">
<id column="b_id" property="bId" jdbcType="INTEGER" />
<result column="b_title" property="bTitle" jdbcType="VARCHAR" />
<result column="b_content" property="bContent" jdbcType="VARCHAR" />
</collection>
</resultMap>

这里的resultMap还多个了一个collection标签,顾名思义这表示一个集合,collection的column表示结果集对应的table的主键。现在再添加一个select标签。

     <select id="selectOneWithBlogs" resultMap="BaseResultMapWithBlogs" parameterType="java.lang.Integer">
select *
from blog b
join user u
on b.user_id=u.id
where u.id = #{id,jdbcType=INTEGER}
</select>

在UserDAO.java里添加一个selectOneWithBlogs()方法。

 package pro.app.inter;
import org.apache.ibatis.annotations.Param; import pro.app.model.User;
import pro.app.model.UserVo; public interface UserDAO {
public User selectOne(@Param("id")Integer id); public void insertOne(User user); public void deleteOne(@Param("id")Integer id); public void updateOne(User user); UserVo selectOneWithBlogs(@Param("id")Integer id);
}

新建一个测试类,观察结果。

 package pro.test;

 import java.io.Reader;

 import org.apache.ibatis.io.Resources;
import org.apache.ibatis.session.SqlSession;
import org.apache.ibatis.session.SqlSessionFactory;
import org.apache.ibatis.session.SqlSessionFactoryBuilder; import pro.app.inter.UserDAO;
import pro.app.model.Blog;
import pro.app.model.UserVo; public class UserVoTest {
private static SqlSessionFactory sqlSessionFactory;
private static Reader reader;
static{
try{
reader= Resources.getResourceAsReader("Configuration.xml");
sqlSessionFactory = new SqlSessionFactoryBuilder().build(reader);
}catch(Exception e){
e.printStackTrace();
}
}
public static SqlSessionFactory getSession(){
return sqlSessionFactory;
}
public static void main(String[] args) {
SqlSession session = sqlSessionFactory.openSession();
try {
UserDAO users = session.getMapper(UserDAO.class);
UserVo user=users.selectOneWithBlogs(1);
for(Blog b:user.getBlogs()){
System.out.println(b.getbTitle());
System.out.println(b.getbContent());
}
} finally {
session.close();
}
}
}

控制台输出

mybatis001

mybatis002

mybatis003

mybatis004

mybatis005

bingo!!!用户对应的博客都被选出来了~

总结

通过resultMap 实现数据的关联。

Hello Mybatis 03 数据关联的更多相关文章

  1. MyBatis框架——mybatis插入数据返回主键(mysql、oracle)

    向数据库中插入数据时,大多数情况都会使用自增列或者UUID做为主键.主键的值都是插入之前无法知道的,但很多情况下我们在插入数据后需要使用刚刚插入数据的主键,比如向两张关联表A.B中插入数据(A的主键是 ...

  2. MyBatis总结-实现关联表查询

    一.一对一关联 1.1.提出需求 根据班级id查询班级信息(带老师的信息) 1.2.创建表和数据 创建一张教师表和班级表,这里我们假设一个老师只负责教一个班,那么老师和班级之间的关系就是一种一对一的关 ...

  3. mybatis框架(6)---mybatis插入数据后获取自增主键

    mybatis插入数据后获取自增主键 首先理解这就话的意思:就是在往数据库表中插入一条数据的同时,返回该条数据在数据库表中的自增主键值. 有什么用呢,举个例子: 你编辑一条新闻,同时需要给该新闻打上标 ...

  4. mybatis插入数据后将其ID返回

    背景 mybatis没有关联保存的功能,所以主从表需要分开保存,这就涉及到主表保存后要再次获取主表ID的环节,以下介绍mybatis插入数据后返回其自增ID的两种方式 方案 1.sql获取 <i ...

  5. 三、mybatis多表关联查询和分布查询

    前言 mybatis多表关联查询和懒查询,这篇文章通过一对一和一对多的实例来展示多表查询.不过需要掌握数据输出的这方面的知识.之前整理过了mybatis入门案例和mybatis数据输出,多表查询是在前 ...

  6. mybatis多对一关联

    mybatis多对一关联查询实现 1.定义实体 定义实体的时候需要注意,若是双向关联,就是说双方的属性中都含有对方对象作为域属性出现, 那么在写toString()方法时需要注意,只让某一方输出即可, ...

  7. EF – 3.EF数据查询基础(下)数据关联

    5.5.1 <关于“数据关联”,你不一定清楚的事> 这讲视频比较全面地介绍了“一对一”.“一对多”和“多对多”三种数据关联类型在关系数据库和Entity Framework数据模型中的实现 ...

  8. 5,SFDC 管理员篇 - 数据模型 - 数据关联

    1,PickList 1,填写基本信息 2, 选择能角色的权限 3,在哪一个层上显示(object 上有多个 Record Type 对应多个层,需要选择在哪一个层显示) 4,Save   2,两个P ...

  9. 微信小程序教学第三章第三节(含视频):小程序中级实战教程:视图与数据关联

    § 视图与数据关联 本文配套视频地址: https://v.qq.com/x/page/z0554wyswib.html 开始前请把 ch3-3 分支中的 code/ 目录导入微信开发工具 首先 首先 ...

随机推荐

  1. Memcached Memcached.ClientLibrary.SockIOPool”的类型初始值设定项引发异常

    又一次遭遇"xxx类型初始值设定项引发异常" 下了个c#实现的轻量级IoC开源项目,可是在本地使用时发现一运行就捕捉到"类型初始值设定项引发异常"的异常信息,调 ...

  2. php发展起源

    PHP原始为Personal Home Page的缩写,已经正式更名为 "PHP: Hypertext Preprocessor".注意不是“Hypertext Preproces ...

  3. ios 的touch事件分析

    IOS之触摸事件和手势   13.1 事件概述 13.2 触摸事件 13.3 手势 13.1 事件概述 事件是当用户手指触击屏幕及在屏幕上移动时,系统不断发送给应用程序的对象. 系统将事件按照特定的路 ...

  4. MyBatis-NET

    http://www.codeproject.com/Articles/894127/MyBatis-NET https://mybatis.github.io/mybatis-3/

  5. php socket解决方案

    最近一直在为移动应用提供 php服务端api,以前 实时交互数据需求不严格(定时从手机端发送http请求),现在业务需求变更, 需要实时交互式接口,必须增加socket. 服务端框架使用YII 1.1 ...

  6. IIS处理并发请求时出现的问题及解决

    一个ASP.NET项目在部署到生产环境时,当用户并发量达到200左右时,IIS出现了明显的请求排队现象,发送的请求都进入等待,无法及时响 应,系统基本处于不可用状态.因经验不足,花了很多时间精力解决这 ...

  7. ROS BY EXAMPLE 1 -- 环境设置与安装

    rbx1 package 的环境设置: sudo apt-get install ros-indigo-turtlebot-bringup \ ros-indigo-turtlebot-create- ...

  8. 51nod 1138 连续整数的和(数学公式)

    1138 连续整数的和 #include <iostream> #include <cmath> #include <cstdio> using namespace ...

  9. VS2015调试UWP程序时提示错误DEP0700 : Registration of the app failed. Another user has already installed

    在同一台windows10电脑上调试过一个工程以后,切换了账号再次调试出现错误 DEP0700 : Registration of the app failed. Another user has a ...

  10. 编译nginx时,编译参数注意点

    --prefix=/usr/local/nginx-1.3.1    有利于统一放置nginx的所有文件,方便管理,强烈建议设置 --with-http_stub_status_module    支 ...