只为成功找方法,不为失败找借口!
MyBatis学习总结(五)——实现关联表查询
一、一对一关联
1.1、提出需求
根据班级id查询班级信息(带老师的信息)
1.2、创建表和数据
创建一张教师表和班级表,这里我们假设一个老师只负责教一个班,那么老师和班级之间的关系就是一种一对一的关系。
复制代码
1 CREATE TABLE teacher(
2 t_id INT PRIMARY KEY AUTO_INCREMENT,
3 t_name VARCHAR(20)
4 );
5 CREATE TABLE class(
6 c_id INT PRIMARY KEY AUTO_INCREMENT,
7 c_name VARCHAR(20),
8 teacher_id INT
9 );
10 ALTER TABLE class ADD CONSTRAINT fk_teacher_id FOREIGN KEY (teacher_id) REFERENCES teacher(t_id);
11
12 INSERT INTO teacher(t_name) VALUES('teacher1');
13 INSERT INTO teacher(t_name) VALUES('teacher2');
14
15 INSERT INTO class(c_name, teacher_id) VALUES('class_a', 1);
16 INSERT INTO class(c_name, teacher_id) VALUES('class_b', 2);
复制代码
表之间的关系如下:
1.3、定义实体类
1、Teacher类,Teacher类是teacher表对应的实体类。
复制代码
1 package me.gacl.domain;
2
3 /**
4 * @author gacl
5 * 定义teacher表对应的实体类
6 */
7 public class Teacher {
8
9 //定义实体类的属性,与teacher表中的字段对应
10 private int id; //id===>t_id
11 private String name; //name===>t_name
12
13 public int getId() {
14 return id;
15 }
16
17 public void setId(int id) {
18 this.id = id;
19 }
20
21 public String getName() {
22 return name;
23 }
24
25 public void setName(String name) {
26 this.name = name;
27 }
28
29 @Override
30 public String toString() {
31 return "Teacher [id=" + id + ", name=" + name + "]";
32 }
33 }
复制代码
2、Classes类,Classes类是class表对应的实体类
复制代码
1 package me.gacl.domain;
2
3 /**
4 * @author gacl
5 * 定义class表对应的实体类
6 */
7 public class Classes {
8
9 //定义实体类的属性,与class表中的字段对应
10 private int id; //id===>c_id
11 private String name; //name===>c_name
12
13 /**
14 * class表中有一个teacher_id字段,所以在Classes类中定义一个teacher属性,
15 * 用于维护teacher和class之间的一对一关系,通过这个teacher属性就可以知道这个班级是由哪个老师负责的
16 */
17 private Teacher teacher;
18
19 public int getId() {
20 return id;
21 }
22
23 public void setId(int id) {
24 this.id = id;
25 }
26
27 public String getName() {
28 return name;
29 }
30
31 public void setName(String name) {
32 this.name = name;
33 }
34
35 public Teacher getTeacher() {
36 return teacher;
37 }
38
39 public void setTeacher(Teacher teacher) {
40 this.teacher = teacher;
41 }
42
43 @Override
44 public String toString() {
45 return "Classes [id=" + id + ", name=" + name + ", teacher=" + teacher+ "]";
46 }
47 }
复制代码
1.4、定义sql映射文件classMapper.xml
复制代码
1
2
3
6
7
8
17
18
23
26
27
28
29
30
31
32
33
34
35
36
41
44
45
46
47
48
49
50
51
54
55
复制代码
在conf.xml文件中注册classMapper.xml
1.5、编写单元测试代码
复制代码
1 package me.gacl.test;
2
3 import me.gacl.domain.Classes;
4 import me.gacl.util.MyBatisUtil;
5 import org.apache.ibatis.session.SqlSession;
6 import org.junit.Test;
7
8 public class Test3 {
9
10 @Test
11 public void testGetClass(){
12 SqlSession sqlSession = MyBatisUtil.getSqlSession();
13 /**
14 * 映射sql的标识字符串,
15 * me.gacl.mapping.classMapper是classMapper.xml文件中mapper标签的namespace属性的值,
16 * getClass是select标签的id属性值,通过select标签的id属性值就可以找到要执行的SQL
17 */
18 String statement = "me.gacl.mapping.classMapper.getClass";//映射sql的标识字符串
19 //执行查询操作,将查询结果自动封装成Classes对象返回
20 Classes clazz = sqlSession.selectOne(statement,1);//查询class表中id为1的记录
21 //使用SqlSession执行完SQL之后需要关闭SqlSession
22 sqlSession.close();
23 System.out.println(clazz);//打印结果:Classes [id=1, name=class_a, teacher=Teacher [id=1, name=teacher1]]
24 }
25
26 @Test
27 public void testGetClass2(){
28 SqlSession sqlSession = MyBatisUtil.getSqlSession();
29 /**
30 * 映射sql的标识字符串,
31 * me.gacl.mapping.classMapper是classMapper.xml文件中mapper标签的namespace属性的值,
32 * getClass2是select标签的id属性值,通过select标签的id属性值就可以找到要执行的SQL
33 */
34 String statement = "me.gacl.mapping.classMapper.getClass2";//映射sql的标识字符串
35 //执行查询操作,将查询结果自动封装成Classes对象返回
36 Classes clazz = sqlSession.selectOne(statement,1);//查询class表中id为1的记录
37 //使用SqlSession执行完SQL之后需要关闭SqlSession
38 sqlSession.close();
39 System.out.println(clazz);//打印结果:Classes [id=1, name=class_a, teacher=Teacher [id=1, name=teacher1]]
40 }
41 }
复制代码
1.6、MyBatis一对一关联查询总结
MyBatis中使用association标签来解决一对一的关联查询,association标签可用的属性如下:
property:对象属性的名称
javaType:对象属性的类型
column:所对应的外键字段名称
select:使用另一个查询封装的结果
二、一对多关联
2.1、提出需求
根据classId查询对应的班级信息,包括学生,老师
2.2、创建表和数据
在上面的一对一关联查询演示中,我们已经创建了班级表和教师表,因此这里再创建一张学生表
复制代码
CREATE TABLE student(
s_id INT PRIMARY KEY AUTO_INCREMENT,
s_name VARCHAR(20),
class_id INT
);
INSERT INTO student(s_name, class_id) VALUES('student_A', 1);
INSERT INTO student(s_name, class_id) VALUES('student_B', 1);
INSERT INTO student(s_name, class_id) VALUES('student_C', 1);
INSERT INTO student(s_name, class_id) VALUES('student_D', 2);
INSERT INTO student(s_name, class_id) VALUES('student_E', 2);
INSERT INTO student(s_name, class_id) VALUES('student_F', 2);
复制代码
2.3、定义实体类
1、Student类
复制代码
1 package me.gacl.domain;
2
3 /**
4 * @author gacl
5 * 定义student表所对应的实体类
6 */
7 public class Student {
8
9 //定义属性,和student表中的字段对应
10 private int id; //id===>s_id
11 private String name; //name===>s_name
12
13 public int getId() {
14 return id;
15 }
16
17 public void setId(int id) {
18 this.id = id;
19 }
20
21 public String getName() {
22 return name;
23 }
24
25 public void setName(String name) {
26 this.name = name;
27 }
28
29 @Override
30 public String toString() {
31 return "Student [id=" + id + ", name=" + name + "]";
32 }
33 }
复制代码
2、修改Classes类,添加一个List
复制代码
1 package me.gacl.domain;
2
3 import java.util.List;
4
5 /**
6 * @author gacl
7 * 定义class表对应的实体类
8 */
9 public class Classes {
10
11 //定义实体类的属性,与class表中的字段对应
12 private int id; //id===>c_id
13 private String name; //name===>c_name
14
15 /**
16 * class表中有一个teacher_id字段,所以在Classes类中定义一个teacher属性,
17 * 用于维护teacher和class之间的一对一关系,通过这个teacher属性就可以知道这个班级是由哪个老师负责的
18 */
19 private Teacher teacher;
20 //使用一个List
21 private List
22
23 public int getId() {
24 return id;
25 }
26
27 public void setId(int id) {
28 this.id = id;
29 }
30
31 public String getName() {
32 return name;
33 }
34
35 public void setName(String name) {
36 this.name = name;
37 }
38
39 public Teacher getTeacher() {
40 return teacher;
41 }
42
43 public void setTeacher(Teacher teacher) {
44 this.teacher = teacher;
45 }
46
47 public List
48 return students;
49 }
50
51 public void setStudents(List
52 this.students = students;
53 }
54
55 @Override
56 public String toString() {
57 return "Classes [id=" + id + ", name=" + name + ", teacher=" + teacher
58 + ", students=" + students + "]";
59 }
60 }
复制代码
2.4、修改sql映射文件classMapper.xml
添加如下的SQL映射信息
复制代码
1
4
8
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
31
34
35
36
37
38
39
40
41
44
45
复制代码
2.5、编写单元测试代码
复制代码
1 package me.gacl.test;
2
3 import me.gacl.domain.Classes;
4 import me.gacl.util.MyBatisUtil;
5 import org.apache.ibatis.session.SqlSession;
6 import org.junit.Test;
7
8 public class Test4 {
9
10 @Test
11 public void testGetClass3(){
12 SqlSession sqlSession = MyBatisUtil.getSqlSession();
13 /**
14 * 映射sql的标识字符串,
15 * me.gacl.mapping.classMapper是classMapper.xml文件中mapper标签的namespace属性的值,
16 * getClass3是select标签的id属性值,通过select标签的id属性值就可以找到要执行的SQL
17 */
18 String statement = "me.gacl.mapping.classMapper.getClass3";//映射sql的标识字符串
19 //执行查询操作,将查询结果自动封装成Classes对象返回
20 Classes clazz = sqlSession.selectOne(statement,1);//查询class表中id为1的记录
21 //使用SqlSession执行完SQL之后需要关闭SqlSession
22 sqlSession.close();
23 //打印结果:Classes [id=1, name=class_a, teacher=Teacher [id=1, name=teacher1], students=[Student [id=1, name=student_A], Student [id=2, name=student_B], Student [id=3, name=student_C]]]
24 System.out.println(clazz);
25 }
26
27 @Test
28 public void testGetClass4(){
29 SqlSession sqlSession = MyBatisUtil.getSqlSession();
30 /**
31 * 映射sql的标识字符串,
32 * me.gacl.mapping.classMapper是classMapper.xml文件中mapper标签的namespace属性的值,
33 * getClass4是select标签的id属性值,通过select标签的id属性值就可以找到要执行的SQL
34 */
35 String statement = "me.gacl.mapping.classMapper.getClass4";//映射sql的标识字符串
36 //执行查询操作,将查询结果自动封装成Classes对象返回
37 Classes clazz = sqlSession.selectOne(statement,1);//查询class表中id为1的记录
38 //使用SqlSession执行完SQL之后需要关闭SqlSession
39 sqlSession.close();
40 //打印结果:Classes [id=1, name=class_a, teacher=Teacher [id=1, name=teacher1], students=[Student [id=1, name=student_A], Student [id=2, name=student_B], Student [id=3, name=student_C]]]
41 System.out.println(clazz);
42 }
43 }
复制代码
2.6、MyBatis一对多关联查询总结
MyBatis中使用collection标签来解决一对多的关联查询,ofType属性指定集合中元素的对象类型。
分类: Mybatis
标签: MyBatis学习总结
好文要顶 关注我 收藏该文
孤傲苍狼
关注 - 75
粉丝 - 2206
+加关注
11 0
(请您对文章做出评价)
« 上一篇:MyBatis学习总结(四)——解决字段名与实体类属性名不相同的冲突
» 下一篇:MyBatis学习总结(六)——调用存储过程
posted on 2015-02-01 16:27 孤傲苍狼 阅读(21458) 评论(8) 编辑 收藏
评论
#1楼 2015-04-13 14:38 ChanShuYi
写得很好,很详细,感谢分享。
支持(0)反对(0)
#2楼 2015-07-22 17:08 to_ny熊
写的不错
支持(0)反对(0)
#3楼 2015-07-31 15:26 落叶秋思
学习ing
支持(0)反对(0)
#4楼 2015-08-07 10:21 礼拜天001
不错 ,正好用到
支持(0)反对(0)
#5楼 2015-08-17 14:39 jkguo
Mark
支持(0)反对(0)
#6楼 2015-09-16 13:08 Laughing_Lz
非常感谢!
支持(0)反对(0)
#7楼 2015-09-29 22:46 Ske.T
如楼主所述:
1
2
3
4
5
6
7
8
这两个情况还是有区别的,
第1种在面对teacher_id == null,的情况下返回的class对象为空,因为sql返回的结果就是查询不到;
但是在第2种情况能够返回带teacher 为null的Class object,其他property还是都有结果的。
当然对应楼主这个范例,teacher 为空没什么意义,但别的情况下就得斟酌了。
支持(0)反对(0)
#8楼 2015-10-20 10:01 BraveSpringer
我自己用mybatis的关联查询时,用的classDao接口来进行查询,获得的是null,但是用sqlsession的selectOne方法就可以获得数据,楼主能不能解答一下
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
dao.ClassDao
public interface ClassDao{
@Select("select c.c_id,c.class_name,c.class_room,c.t_id,c.cap,t.teacher_name,t.subject from class c,teacher t where c.t_id=t.id and c.c_id=#{id}")
public Class getClassById(Integer id);
}
mapping.ClassMapper
//测试
ClassDaoTest
@Test
public void testGetClassById() {
try{
SqlSession sqlsession=MyBatisUtil.getSqlSession(true);
//基于注解的关联查询尚有问题
ClassDao cd=sqlsession.getMapper(ClassDao.class);
Class cls=new Class();
//用session的selectOne方法可以获得数据 //cls=sqlsession.selectOne("mapping.ClassMapper.getClassById",1);
cls=cd.getClassById(3);
System.out.println("classdao"+cls.toString()+"; "+cls.getId());
}
catch(Exception ex){
ex.printStackTrace();
}
//打印相关sql语句