MyBatis的association示例——MyBatis学习笔记之三
前两篇博文介绍的都是单表映射,而实际上很多时候我们需要用到较复杂的映射。今天学会的association的用法,就是一例,现写出来和大家分享(为简洁起见,ant工程中各文件、目录的布局,以及其它与前面的例子重复的内容,将不再赘述。以后博文亦将如此)。
假设每个学生都有一名指导老师,本示例的任务就是查询出学生的详细信息,这就包括学生的指导教师的信息。为此,应先增加一个教师的实体类。如下所示(和以前一样的原因,读者不要复制本文的配置文件。在文章下方的附件下载处,可下载本示例的完整代码):
package com.abc.domain;
public class Teacher{
private int id;
private String name; //姓名
private String gender;//性别
private String researchArea;//研究方向
private String title;//职称
public int getId() {
return id;
}
public void setId(int id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getGender() {
return gender;
}
public void setGender(String gender) {
this.gender = gender;
}
public String getResearchArea() {
return researchArea;
}
public void setResearchArea(String researchArea) {
this.researchArea = researchArea;
}
public String getTitle() {
return title;
}
public void setTitle(String title) {
this.title = title;
}
}
相应地,在数据库中,应增加教师表。完成此任务的脚本(teacher.sql)如下:
/*数据库编码UTF8,以下命令是为了在脚本和
命令行中支持中文*/
set names gbk;
/*切换到courseman数据库*/
use courseman;
/*创建成绩表*/
drop table if exists teacher;
CREATE TABLE teacher(
id int NOT NULL AUTO_INCREMENT primarykey,
name varchar(10) NOT NULL,/*姓名*/
gender char(1) NOT NULL,/*性别*/
research_area varchar(20) NOT NULL,/*研究方向*/
title varchar(6) NOT NULL/*职称*/
);
/*添加第一条记录,自动生成的ID为1*/
insert into teacher(name,gender,research_area, title)
values('张伟','男','软件工程','讲师');
在命令行下以courseman身份登录MySQL(mysql –ucourseman –pabc123),用source命令运行此脚本(若 teacher.sql放在D盘,即d:\ teacher.sql,则运行命令source d:/teacher.sql。注意,这里是“/”,而不 是“\”。以下脚本的运行方式不再赘述)。
接着为学生增加指导教师属性,如下:
private Teacher supervisor; //指导教师
并为此属性编写getter和setter方法,此处略去。
相应地,为学生表增加一个指导教师ID的字段,脚本(supervisor.sql)如下:
/*切换到courseman数据库*/
use courseman;
/*为学生表添加指导老师ID列*/
alter TABLE student add supervisor_id int not null
references teacher(id);
/*把上面新增的教师作为目前学生的指导教师*/
update student set supervisor_id=1;
在MyBatis的核心配置文件configuration.xml中增加教师类型的别名定义,如下:
<typeAliases>
<typeAlias alias="Student" type="com.abc.domain.Student"/>
<!--增加的教师类型别名-->
<typeAlias alias="Teacher" type="com.abc.domain.Teacher"/>
</typeAliases>
然后,我们需要修改StudentMapper.xml中的select语句及要用到的resultMap元素,这是本示例的关键部分。
首先,为了能够同时查询到学生的指导教师的信息,修改select语句如下:
<select id="getById" parameterType="int" resultMap="studentResultMap">
select st.id,st.name,st.gender,
st.major,st.grade,
<!--为教师的id取别名,避免MyBatis向教师实体注入
此属性时与学生id混淆。以下的name和gender等属性
也是如此-->
t.id t_id,t.name t_name,t.gender t_gender,
t.title,t.research_area
from student st, teacher t
where st.supervisor_id = t.id
and st.id=#{id}
</select>
为了实现查询结果与实体的映射,需要修改resultMap元素。此时的学生实体拥有一个指导教师属性(supervisor),而该属性本身就是一个实体。这是一种has-a关系,亦即一个学生有一个指导教师,而association元素就是处理这种关系的映射的。我们为resultMap添加 association如下(从第11行开始):
<resultMap id="studentResultMap" type="Student">
<!--普通属性映射与以前一致-->
<id property="id" column="id"/>
<result property="name" column="name"/>
<result property="gender" column="gender"/>
<result property="major" column="major"/>
<result property="grade" column="grade"/>
<!--property="supervisor"表明这是为了映射学生实体的
supervisor属性。javaType="Teacher"用到了Teacher这个
别名定义,并指出了supervisor属性的java类型-->
<association property="supervisor" javaType="Teacher">
<!--教师自身的属性与数据库字段的映射。注意这里用到了字段别名-->
<id property="id" column="t_id"/>
<result property="name" column="t_name"/>
<result property="gender" column="t_gender"/>
<result property="researchArea" column="research_area"/>
<result property="title" column="title"/>
</association>
</resultMap>
其中的javaType属性为必须,否则报以下错误:
这次的执行类是AssociationDemo,代码如下:
package com.demo;
import org.apache.ibatis.session.SqlSessionFactory;
import org.apache.ibatis.session.SqlSession;
import com.abc.mapper.StudentMapper;
import com.abc.domain.Student;
import com.util.SqlSessionFactoryGen;
public class AssociationDemo
{
//获取SqlSessionFactory实例
private static SqlSessionFactory factory
= SqlSessionFactoryGen.getSqlSessionFactory();
public static void main(String[] args)
{
SqlSession session = factory.openSession();
StudentMapper mapper =
session.getMapper(StudentMapper.class);
//笔者的数据库中只有ID为4的学生。读者若运行此程序,
//须使用你的数据库中存在的学生ID。否则报空指针异常
Student student = mapper.getById(4);
//使用StringBuilder的append操作代替字符串的“+”
//操作可提高执行效率
StringBuilder sb = new StringBuilder("学生信息:\n");
sb.append("姓名:");
sb.append(student.getName());
sb.append(" ");
sb.append( "专业:");
sb.append(student.getMajor());
sb.append(" 年级:");
sb.append(student.getGrade());
sb.append("\n");
sb.append("指导教师信息:\n");
sb.append("姓名:");
sb.append(student.getSupervisor().getName());
sb.append(" ");
sb.append("职称:");
sb.append(student.getSupervisor().getTitle());
sb.append(" ");
sb.append("研究方向:");
sb.append(student.getSupervisor().getResearchArea());
System.out.println(sb.toString());
session.close();
}
}
相应地,应修改ant的生成文件build.xml中的run target,指定这个类是要运行的类:
<target name="run" depends="compile">
<!--指定AssociationDemo为要运行的类-->
<java fork="true" classname="com.demo.AssociationDemo" classpathref="library">
<classpath path="${targetdir}"/>
</java>
</target>
执行结果如下:
MyBatis的association示例——MyBatis学习笔记之三的更多相关文章
- java maven、springmvc、mybatis 搭建简单Web项目学习笔记
前言: 空余的时间,学学 Java,没准哪天用的到: 环境搭建折腾了好几天,总算搞顺了,也做个学习笔记,以防后面会忘记: 一.安装文件及介绍 JDK:jdk1.8.0 77 eclipse-maven ...
- Django 学习笔记之三 数据库输入数据
假设建立了django_blog项目,建立blog的app ,在models.py里面增加了Blog类,同步数据库,并且建立了对应的表.具体的参照Django 学习笔记之二的相关命令. 那么这篇主要介 ...
- 【Visual C++】游戏编程学习笔记之三:游戏循环的使用
本系列文章由@二货梦想家张程 所写,转载请注明出处. 本文章链接:http://blog.csdn.net/terence1212/article/details/44208419 作者:Zee ...
- hive学习笔记之三:内部表和外部表
欢迎访问我的GitHub https://github.com/zq2599/blog_demos 内容:所有原创文章分类汇总及配套源码,涉及Java.Docker.Kubernetes.DevOPS ...
- OpenGL学习笔记 之三 (简单示例 太阳月亮地球)
#include<glut.h> // 太阳.地球和月亮 // 假设每个月都是30天 // 一年12个月,共是360天 ;//day的变化:从0到359 void myDisplay(vo ...
- 【Java学习笔记之三十一】详解Java8 lambda表达式
Java 8 发布日期是2014年3月18日,这次开创性的发布在Java社区引发了不少讨论,并让大家感到激动.特性之一便是随同发布的lambda表达式,它将允许我们将行为传到函数里.在Java 8之前 ...
- 学习笔记之三十年软件开发之路 - Things I Learnt The Hard Way (in 30 Years of Software Development)
三十年软件开发之路 https://mp.weixin.qq.com/s/EgN-9bIHonRid1DM0csQDw https://blog.juliobiason.net/thoughts/th ...
- Vue.js 学习笔记之三:与服务器的数据交互
显而易见的,之前的02_toDoList存在着一个很致命的缺陷.那就是它的数据只存在于浏览器端,一但用户关闭或重新载入页面,他之前加入到程序中的数据就会全部丢失,一切又恢复到程序的初始状态.要想解决这 ...
- bootstrap学习笔记之三(组件的使用)
bootstrap组件需要引入bootstrap.js才行,当然要引入bootstrap.js首先得引入JQuery. 一.下拉菜单 将下拉菜单触发器和下拉菜单都包裹在 .dropdown 里,或者另 ...
随机推荐
- POJ 1060 Modular multiplication of polynomials(多项式的加减乘除,除法转化成减法来求)
题意:给出f(x),g(x),h(x)的 (最高次幂+1)的值,以及它们的各项系数,求f(x)*g(x)/h(x)的余数. 这里多项式的系数只有1或0,因为题目要求:这里多项式的加减法是将系数相加/减 ...
- orale内置函数COALESCE和GREATEST和LEAST
1. COALESCE 返回该表达式列表的第一个非空value. 格式: COALESCE(value1, value2, value3, ...) 含义: 返回value列表第一个非空的值. val ...
- 青鸟 王云鹏老师写的SqlHelper 泛型方法,反射,支持实体类
1: using System; 2: using System.Collections.Generic; 3: using System.Linq; 4: using System.Text; 5: ...
- POJ3267The Cow Lexicon
http://poj.org/problem?id=3267 题意 : 给你一个message,是给定字符串,然后再给你字典,让你将message与字典中的单词进行匹配,输出要删掉多少字母. 思路 : ...
- IText 生成页脚页码
做doc文档报表的时候可能遇到这样的需求: 每一个页面需要页码,用IText可以完成这样的需求. IText生成doc文档需要三个包:iTextAsian.jar,iText-rtf-2.1.4.ja ...
- Python中itertools模块
itertools模块包含创建有效迭代器的函数,可以用各种方式对数据进行循环操作,此模块中的所有函数返回的迭代器都可以与for循环语句以及其他包含迭代器(如生成器和生成器表达式)的函数联合使用. ch ...
- AndroidJNI 调用JAVA(转)
转自:http://www.cnblogs.com/likwo/archive/2012/05/21/2512400.html 1. JNIEnv对象 对于本地函数 JNIEXPORT ...
- VS2005控制台程序修改nb0文件
VS2005控制台程序修改nb0文件 我们要实现的功能就是通过CMD传递进来的值来在nb0文件末尾增加版本信息,新建控制台程序,自动生成的main函数如下,默认的代码非常简单: int _tmain( ...
- 面试题_103_to_124_关于 OOP 和设计模式的面试题
这部分包含 Java 面试过程中关于 SOLID 的设计原则,OOP 基础,如类,对象,接口,继承,多态,封装,抽象以及更高级的一些概念,如组合.聚合及关联.也包含了 GOF 设计模式的问题. 103 ...
- [LeetCode#247] Strobogrammatic Number II
Problem: A strobogrammatic number is a number that looks the same when rotated 180 degrees (looked a ...