MyBatis的association示例——MyBatis学习笔记之三

 前两篇博文介绍的都是单表映射,而实际上很多时候我们需要用到较复杂的映射。今天学会的association的用法,就是一例,现写出来和大家分享(为简洁起见,ant工程中各文件、目录的布局,以及其它与前面的例子重复的内容,将不再赘述。以后博文亦将如此)。
      假设每个学生都有一名指导老师,本示例的任务就是查询出学生的详细信息,这就包括学生的指导教师的信息。为此,应先增加一个教师的实体类。如下所示(和以前一样的原因,读者不要复制本文的配置文件。在文章下方的附件下载处,可下载本示例的完整代码):
 
 
  1. package com.abc.domain;  

  2. publicclass Teacher{  

  3. privateint id;  

  4. private String name;  //姓名 

  5. private String gender;//性别 

  6. private String researchArea;//研究方向 

  7. private String title;//职称 

  8. publicint getId() {  

  9. return id;  

  10.     }  

  11. publicvoid setId(int id) {  

  12. this.id = id;  

  13.     }  

  14. public String getName() {  

  15. return name;  

  16.     }  

  17. publicvoid setName(String name) {  

  18. this.name = name;  

  19.     }  

  20. public String getGender() {  

  21. return gender;  

  22.     }  

  23. publicvoid setGender(String gender) {  

  24. this.gender = gender;  

  25.     }  

  26. public String getResearchArea() {  

  27. return researchArea;  

  28.     }  

  29. publicvoid setResearchArea(String researchArea) {  

  30. this.researchArea = researchArea;  

  31.     }  

  32. public String getTitle() {  

  33. return title;  

  34.     }  

  35. publicvoid setTitle(String title) {  

  36. this.title = title;  

  37.     }  

      相应地,在数据库中,应增加教师表。完成此任务的脚本(teacher.sql)如下:
 
 
  1. /*数据库编码UTF8,以下命令是为了在脚本和  

  2. 命令行中支持中文*/  

  3. set names gbk;  

  4. /*切换到courseman数据库*/  

  5. use courseman;  

  6. /*创建成绩表*/  

  7. droptable if exists teacher;  

  8. CREATETABLE teacher(  

  9.    id intNOTNULL AUTO_INCREMENT primarykey,  

  10. namevarchar(10) NOTNULL,/*姓名*/  

  11.    gender char(1) NOTNULL,/*性别*/  

  12.    research_area varchar(20) NOTNULL,/*研究方向*/  

  13.    title varchar(6) NOTNULL/*职称*/    

  14. );  

  15. /*添加第一条记录,自动生成的ID为1*/  

  16. insertinto teacher(name,gender,research_area, title)  

  17. values('张伟','男','软件工程','讲师'); 

       在命令行下以courseman身份登录MySQL(mysql –ucourseman –pabc123),用source命令运行此脚本(若teacher.sql放在D盘,即d:\ teacher.sql,则运行命令source d:/teacher.sql。注意,这里是“/”,而不是“\”。以下脚本的运行方式不再赘述)。

      接着为学生增加指导教师属性,如下:
 
 
  1. private Teacher supervisor; //指导教师

       并为此属性编写getter和setter方法,此处略去。

      相应地,为学生表增加一个指导教师ID的字段,脚本(supervisor.sql)如下: 
 
 
  1. /*切换到courseman数据库*/  

  2. use courseman;  

  3. /*为学生表添加指导老师ID列*/  

  4. alterTABLE student add supervisor_id intnotnull

  5. references teacher(id);  

  6. /*把上面新增的教师作为目前学生的指导教师*/  

  7. update student set supervisor_id=1; 

       在MyBatis的核心配置文件configuration.xml中增加教师类型的别名定义,如下: 

 
 
  1. <typeAliases>

  2. <typeAliasalias="Student"

  3. type="com.abc.domain.Student"/>

  4. <!--增加的教师类型别名-->

  5. <typeAliasalias="Teacher"

  6. type="com.abc.domain.Teacher"/>

  7. </typeAliases>

       然后,我们需要修改StudentMapper.xml中的select语句及要用到的resultMap元素,这是本示例的关键部分。

     首先,为了能够同时查询到学生的指导教师的信息,修改select语句如下: 
 
 
  1. <selectid="getById"parameterType="int"resultMap="studentResultMap">

  2.           select st.id,st.name,st.gender,  

  3.           st.major,st.grade,  

  4.           <!--为教师的id取别名,避免MyBatis向教师实体注入  

  5.           此属性时与学生id混淆。以下的name和gender等属性  

  6.           也是如此-->

  7.           t.id t_id,t.name t_name,t.gender t_gender,  

  8.           t.title,t.research_area  

  9.           from student st, teacher t  

  10.           where st.supervisor_id = t.id  

  11.           and st.id=#{id}  

  12. </select>

       为了实现查询结果与实体的映射,需要修改resultMap元素。此时的学生实体拥有一个指导教师属性(supervisor),而该属性本身就是一个实体。这是一种has-a关系,亦即一个学生有一个指导教师,而association元素就是处理这种关系的映射的。我们为resultMap添加association如下(从第12行开始): 

 
 
  1. <resultMap id="studentResultMap"type="Student">

  2. <!--普通属性映射与以前一致-->

  3. <id property="id"column="id"/>

  4. <result property="name"column="name"/>

  5. <result property="gender"column="gender"/>

  6. <result property="major"column="major"/>

  7. <result property="grade"column="grade"/>

  8.           <!--property="supervisor"表明这是为了映射学生实体的  

  9.           supervisor属性。javaType="Teacher"用到了Teacher这个  

  10.           别名定义,并指出了supervisor属性的java类型-->

  11. <association property="supervisor"javaType="Teacher">

  12.              <!--教师自身的属性与数据库字段的映射。注意这里  

  13.              用到了字段别名-->

  14. <id property="id"column="t_id"/>

  15. <result property="name"column="t_name"/>

  16. <result property="gender"column="t_gender"/>

  17. <result property="researchArea"column="research_area"/>

  18. <result property="title"column="title"/>

  19. </association>

  20. </resultMap>

       其中的javaType属性为必须,否则报以下错误: 

       这次的执行类是AssociationDemo,代码如下: 

 
 
  1. package com.demo;  

  2. import org.apache.ibatis.session.SqlSessionFactory;  

  3. import org.apache.ibatis.session.SqlSession;  

  4. import com.abc.mapper.StudentMapper;  

  5. import com.abc.domain.Student;  

  6. import com.util.SqlSessionFactoryGen;  

  7. publicclass AssociationDemo  

  8. {  

  9. //获取SqlSessionFactory实例 

  10. privatestatic SqlSessionFactory factory  

  11.         = SqlSessionFactoryGen.getSqlSessionFactory();  

  12. publicstaticvoid main(String[] args)  

  13.     {  

  14.         SqlSession session = factory.openSession();  

  15.         StudentMapper mapper =   

  16.                session.getMapper(StudentMapper.class);  

  17. //笔者的数据库中只有ID为4的学生。读者若运行此程序, 

  18. //须使用你的数据库中存在的学生ID。否则报空指针异常 

  19.         Student student = mapper.getById(4);  

  20. //使用StringBuilder的append操作代替字符串的“+” 

  21. //操作可提高执行效率 

  22.         StringBuilder sb = new StringBuilder("学生信息:\n");  

  23.         sb.append("姓名:");  

  24.         sb.append(student.getName());  

  25.         sb.append(" ");  

  26.         sb.append( "专业:");  

  27.         sb.append(student.getMajor());  

  28.         sb.append(" 年级:");  

  29.         sb.append(student.getGrade());  

  30.         sb.append("\n");  

  31.         sb.append("指导教师信息:\n");  

  32.         sb.append("姓名:");  

  33.         sb.append(student.getSupervisor().getName());  

  34.         sb.append(" ");  

  35.         sb.append("职称:");  

  36.         sb.append(student.getSupervisor().getTitle());  

  37.         sb.append(" ");  

  38.         sb.append("研究方向:");  

  39.         sb.append(student.getSupervisor().getResearchArea());  

  40.         System.out.println(sb.toString());  

  41.         session.close();  

  42.     }  

       相应地,应修改ant的生成文件build.xml中的run target,指定这个类是要运行的类:

 
 
  1. <targetname="run"depends="compile">

  2. <!--指定AssociationDemo为要运行的类-->

  3. <javafork="true"classname="com.demo.AssociationDemo"

  4. classpathref="library">

  5. <classpathpath="${targetdir}"/>

  6. </java>

  7. </target>

      执行结果如下:

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值