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/ 目录导入微信开发工具 首先 首先 ...
随机推荐
- webpack+react配置
$ npm install -g webpack $ npm install -g webpack-dev-server如果遇到类似 EACESS 错误,则需要用超级用户的模式运行 $ sudo np ...
- 【日常小问题】windows系统操作技巧
如何将文件夹视图设置为“详细信息” @2015.3.30 这段时间刚换了电脑,资源管理器视图模式为“大图标”,很不符合俺的操作习惯.修改之—— 1.打开一个文件夹,设置为详细信息显示. 2.点击工具- ...
- 倒计时(jQuery)
倒计时,在网页制作中会经常用到,特别是一些节日活动页面运用更是广泛.今天就和大家分享一下jQuery如何实现倒计时.话不多说,言归正传,倒计时开始: 首先,既然是用jQuery实现,那么前提就是要先引 ...
- 【ImageView】ImageView点击事件报错空指针
今天在使用自定义圆形imageview的时候,想利用其点击事件来实现查看个人资料功能,但是该空间在Activity中的onCreate方法中调用点击事件总是出现空指针异常,每次程序都进不去主页面,到处 ...
- 分析sql语句所有表名及其别名的正则表达式
最近为了 写一个分布式的数据组件构想了很多的方案,最近一个简单易行的方案终于在脑袋里成型.昨晚想到凌晨1点多,发现方案虽简单,但所有的数据库工具就不能使用了 .除非自己写一下查询分析器来执行程序员自己 ...
- 使用OFBIZ 时,使用的键入提示。
对商品的键入提示 ,效果如图(当输入关键字时,会提示出相应的数据) 首先要引入相应的插件 页面字段 js方法
- arduino 入手
新买了个ardhuino 入门套件 1. Win7 Driver issue: Need to install the below drivers, because my windows is ins ...
- java面向对象(封装-继承-多态)
框架图 理解面向对象 面向对象是相对面向过程而言 面向对象和面向过程都是一种思想 面向过程强调的是功能行为 面向对象将功能封装进对象,强调具备了功能的对象. 面向对象是基于面向过程的. 面向对象的特点 ...
- DateTime与DateTime?赋值问题以及null类型的如何赋值问题
解决方案: //主要用到向下兼容原理,DateTime?继承于DateTime: string req = "为字符串的参数"; DateTime? dt = null; Date ...
- 跳转页面,传递参数——android
android 跳转页面并传递对象(实体类)——项目中是集港收货类 网上资料:两种传递方法Serializable,parcelable 优劣比较:Serializable数据更持久化,网络传输或数据 ...