Hello Mybatis 03 数据关联
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 数据关联的更多相关文章
- MyBatis框架——mybatis插入数据返回主键(mysql、oracle)
向数据库中插入数据时,大多数情况都会使用自增列或者UUID做为主键.主键的值都是插入之前无法知道的,但很多情况下我们在插入数据后需要使用刚刚插入数据的主键,比如向两张关联表A.B中插入数据(A的主键是 ...
- MyBatis总结-实现关联表查询
一.一对一关联 1.1.提出需求 根据班级id查询班级信息(带老师的信息) 1.2.创建表和数据 创建一张教师表和班级表,这里我们假设一个老师只负责教一个班,那么老师和班级之间的关系就是一种一对一的关 ...
- mybatis框架(6)---mybatis插入数据后获取自增主键
mybatis插入数据后获取自增主键 首先理解这就话的意思:就是在往数据库表中插入一条数据的同时,返回该条数据在数据库表中的自增主键值. 有什么用呢,举个例子: 你编辑一条新闻,同时需要给该新闻打上标 ...
- mybatis插入数据后将其ID返回
背景 mybatis没有关联保存的功能,所以主从表需要分开保存,这就涉及到主表保存后要再次获取主表ID的环节,以下介绍mybatis插入数据后返回其自增ID的两种方式 方案 1.sql获取 <i ...
- 三、mybatis多表关联查询和分布查询
前言 mybatis多表关联查询和懒查询,这篇文章通过一对一和一对多的实例来展示多表查询.不过需要掌握数据输出的这方面的知识.之前整理过了mybatis入门案例和mybatis数据输出,多表查询是在前 ...
- mybatis多对一关联
mybatis多对一关联查询实现 1.定义实体 定义实体的时候需要注意,若是双向关联,就是说双方的属性中都含有对方对象作为域属性出现, 那么在写toString()方法时需要注意,只让某一方输出即可, ...
- EF – 3.EF数据查询基础(下)数据关联
5.5.1 <关于“数据关联”,你不一定清楚的事> 这讲视频比较全面地介绍了“一对一”.“一对多”和“多对多”三种数据关联类型在关系数据库和Entity Framework数据模型中的实现 ...
- 5,SFDC 管理员篇 - 数据模型 - 数据关联
1,PickList 1,填写基本信息 2, 选择能角色的权限 3,在哪一个层上显示(object 上有多个 Record Type 对应多个层,需要选择在哪一个层显示) 4,Save 2,两个P ...
- 微信小程序教学第三章第三节(含视频):小程序中级实战教程:视图与数据关联
§ 视图与数据关联 本文配套视频地址: https://v.qq.com/x/page/z0554wyswib.html 开始前请把 ch3-3 分支中的 code/ 目录导入微信开发工具 首先 首先 ...
随机推荐
- 关于公司VPN专线接入的一个案例
最近互通网络技术部门收到一到封邮件,是关于公司VPN接入的问题,问题如下: 条件:A区为办公地点1,其他办公电脑20台,并通过写字楼的宽带接口联通外网. B区为新办公地点2,预设应用.数据库.备份服务 ...
- Android视图状态及重绘流程分析,带你一步步深入了解View(三)
转载请注明出处:http://blog.csdn.net/guolin_blog/article/details/17045157 在前面一篇文章中,我带着大家一起从源码的层面上分析了视图的绘制流程, ...
- python ImportError: DLL load failed: %1 不是有效的 Win32 应用程序
导入的时候报出了 ImportError 在windows上安装python 的模块后,导入模块时报 python ImportError: DLL load failed: %1 不是有效的 Win ...
- 设计模式可复用面向对象软件设计基础之对象创建型模式—ABSTRACT FACTORY( 抽象工厂)
意图 提供一个创建一系列相关或相互依赖对象的接口,而无需指定它们具体的类. 适用性 在以下情况可以使用 Abstract Factory模式 • 一个系统要独立于它的产品的创建.组合和表示时. • 一 ...
- 使用hibernate可以优化的地方
a. 在查询字符串中,应该总是使用jdbc的占位符?,或使用使用命名参数:,不要自查询中使用字符串值来代替非常量值. b. Flush会影响性能,频繁刷新影响性能,尽量减少不必要的刷新. c. ...
- window对象常用方法
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...
- GERBER文件
GERBER文件 GERBER文件是一种国际标准的光绘格式文件,它包含RS-274-D和RS-274-X两种格式,其中RS-274-D称为基本GERBER格式,并 要同时附带D码文件才能完整描述一张图 ...
- Android应用:StatusBar状态栏、NavigationBar虚拟按键栏、ActionBar标题栏、Window屏幕内容区域等的宽高
一.屏幕中各种栏目以及屏幕的尺寸 当我们需要计算屏幕中一些元素的高度时,或许需要先获取到屏幕或者各种栏目的高度,下面这个类包含了Status bar状态栏,Navigation bar虚拟按键栏,Ac ...
- jquery 监听input输入值事件
<html> <head> <title></title> <script type="text/javascript" sr ...
- JS动态级联菜单
JS动态级联菜单是前端常用的一个功能,特此抽时间研究了下,附上代码 <html> <head> <meta charset="utf-8" /> ...