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>

执行结果如下:

时间: 2024-08-02 09:21:28

MyBatis的association示例——MyBatis学习笔记之三的相关文章

MySQL学习笔记之三 表类型

你能用的数据库引擎取决于MySQL在安装时候是如何被编译的.要添加一个新的引擎,就必须编译MySQL.仅仅为了添加一个特性而编译应用程序的想法对于Windows的开发人员来说可能有点小题大做,得不偿失,但是在Unix的世界里,这已经成为了标准.在缺省的情况下,MySQL支持三个引擎:ISAM.MyISAM和HEAP.另外两种类型InnoDB和Berkley(BDB),也常常可以使用. ISAM ISAM是一个定义明确且历经时间考验的数据表格管理方法,它在设计之初就考虑到数据库被查询的次数远远大于

Citrix XenMobile学习笔记之三:MAM移动应用管理(Mobility Application Management)

产品简介 思杰(Citrix)在其全面的企业移动解决方案中提供了企业级移动应用管理(MAM)功能.XenMobile MAM 版由CloudGateway发展而来.CloudGateway是思杰进入MAM的跳板.该产品的所有功能在XenMobile的MAM版中都有,现在叫做App版,思杰还有带有完整功能的企业版.XenMobile的移动应用管理组件运行在iOS.安卓.Windows.Windows Phone.Mac OS X.黑莓甚至塞班上.XenMobile支持iOS与安卓上的原生应用.MA

《Hibernate学习笔记之三》:联合主键的映射

<Hibernate学习笔记之三>:联合主键的映射 就如在前面所举的例子一样,是使用的id作为唯一的主键,一般情况下我们也只使用唯一的一个属性作为主键,但是在实际中,我们可能会遇到几个属性作为主键的情况,因此,在本篇博文中,就来介绍下,联合主键的映射关系应该如何来做?? 联合主键的映射有两种方式来进行实现. 1.使用映射文件 XXX.bhm.xml 2.使用Annotation Hibernate首先需要使用联合主键的实体类必须实现Serializable接口,即为了使序列能够被序列化进行传输

马哥Linux学习笔记之三——加密

1.明文传输 ftp,http,smtp,telnet 2.机密性:plaintext-->转换规则-->ciphertext ciphertext-->转换规则-->plaintext 完整性:单项加密算法,提取数据特征码.输入一样,输出必然一样:雪崩效应,输入的微小改变,将会引起结果的巨大改变:无论原始数据是多少,结果大小都是相同的:不可逆,无法根据特征码还原原来的数据. 3.密钥 4.对称加密:有加密算法,有密钥 5.密钥交换算法 IKE(Internet Key Excha

Django 学习笔记之三 数据库输入数据

假设建立了django_blog项目,建立blog的app ,在models.py里面增加了Blog类,同步数据库,并且建立了对应的表.具体的参照Django 学习笔记之二的相关命令. 那么这篇主要介绍往数据库中添加数据,django默认的是sqlite3数据库. 在建立完django_blog项目后,不要忘了把 blog 加入到 settings.py 中的 INSTALLED_APPS 中. 一.同步数据库,创建相应的表 具体的参照Django 学习笔记之二的的同步数据库. 二.数据输入 下

Kinect学习笔记之三Kinect开发环境配置详解

0.前言: 首先说一下我的开发环境,Visual Studio是2013的,系统是win8的64位版本,SDK是Kinect for windows SDK 1.8版本.虽然前一篇博文费了半天劲,翻译了2.0SDK的新特性,但我还是决定要回退一个版本. 其实我之前一直在用2.0的SDK在调试Kinect,但无奈实验室提供的Kinect是for Windows 1.0版本的,而且Kinect从1.8之后就好像是一个分水岭,就比如win8和win7有很大的差别,2.0版的Kinect和SDK都是相较

hibernate学习笔记之三(一级缓存的使用和管理)

(hibernate学习笔记系列来自于 "疯狂Java" Hibernate视频学习) 一级缓存 一级缓存生命周期很短,与session生命周期一致,所以一级缓存也叫session级缓存或事务级缓存.位于缓存中的对象处于持久化状态,它与表中的相关记录对应,session能够在某些时间点,按照缓存中持久化对象的属性来同步数据库中表的记录,这一过程称为清理缓存. 一级缓存实现原理. session缓存是由它的实现类sessionImpl中定义的一些集合属性构成的,原理是保证有一个引用在关联

hibernate学习笔记之三(单条记录的增删改查)

(hibernate学习笔记系列来自于 "疯狂Java" Hibernate视频学习) 保存用户 session.save(person); session.persist(person); 查询用户 load(Class theClass,Serializable id) get(Class theClass,Serializable id) 修改用户 session.update(person) session.merger(person); 保存或修改用户 session.sav

OpenGL学习笔记 之三 (简单示例 太阳月亮地球)

#include<glut.h> // 太阳.地球和月亮 // 假设每个月都是30天 // 一年12个月,共是360天 static int day = 150;//day的变化:从0到359 void myDisplay(void) { glDepthFunc(GL_LEQUAL);//设置深度<=通过,与物体之间的前后有关 glEnable(GL_DEPTH_TEST);//深度检测 glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT)