MyBatis05:一對多和多對一處理
多對一的處理
多對一的了解:
- 多個學生對應一個老師
- 如果對于學生這邊,就是一個多對一的現象,即從學生這邊關聯一個老師!
資料庫設計
MyBatis05:一對多和多對一處理
CREATE TABLE `teacher` (
`id` INT(10) NOT NULL,
`name` VARCHAR(30) DEFAULT NULL,
PRIMARY KEY (`id`)
) ENGINE=INNODB DEFAULT CHARSET=utf8
INSERT INTO teacher(`id`, `name`) VALUES (1, '蕭老師');
CREATE TABLE `student` (
`id` INT(10) NOT NULL,
`name` VARCHAR(30) DEFAULT NULL,
`tid` INT(10) DEFAULT NULL,
PRIMARY KEY (`id`),
KEY `fktid` (`tid`),
CONSTRAINT `fktid` FOREIGN KEY (`tid`) REFERENCES `teacher` (`id`)
) ENGINE=INNODB DEFAULT CHARSET=utf8
INSERT INTO `student` (`id`, `name`, `tid`) VALUES ('1', '小明', '1');
INSERT INTO `student` (`id`, `name`, `tid`) VALUES ('2', '小紅', '1');
INSERT INTO `student` (`id`, `name`, `tid`) VALUES ('3', '小張', '1');
INSERT INTO `student` (`id`, `name`, `tid`) VALUES ('4', '小李', '1');
INSERT INTO `student` (`id`, `name`, `tid`) VALUES ('5', '小王', '1');
搭建測試環境
1、IDEA安裝Lombok插件
2、引入Maven依賴
<!-- https://mvnrepository.com/artifact/org.projectlombok/lombok -->
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
<version>1.16.10</version>
</dependency>
3、在代碼中增加注解
@Data //GET,SET,ToString,有參,無參構造
public class Teacher {
private int id;
private String name;
}
@Data
public class Student {
private int id;
private String name;
//多個學生可以是同一個老師,即多對一
private Teacher teacher;
}
4、編寫實體類對應的Mapper接口 【兩個】
- 無論有沒有需求,都應該寫上,以備後來之需!
public interface StudentMapper {
}
public interface TeacherMapper {
}
5、編寫Mapper接口對應的 mapper.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="com.kuang.mapper.StudentMapper">
</mapper>
<?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="com.kuang.mapper.TeacherMapper">
</mapper>
按查詢嵌套處理
1、給StudentMapper接口增加方法
//擷取所有學生及對應老師的資訊
public List<Student> getStudents();
2、編寫對應的Mapper檔案
<?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="com.kuang.mapper.StudentMapper">
<!--
需求:擷取所有學生及對應老師的資訊
思路:
1. 擷取所有學生的資訊
2. 根據擷取的學生資訊的老師ID->擷取該老師的資訊
3. 思考問題,這樣學生的結果集中應該包含老師,該如何處理呢,資料庫中我們一般使用關聯查詢?
1. 做一個結果集映射:StudentTeacher
2. StudentTeacher結果集的類型為 Student
3. 學生中老師的屬性為teacher,對應資料庫中為tid。
多個 [1,...)學生關聯一個老師=> 一對一,一對多
4. 檢視官網找到:association – 一個複雜類型的關聯;使用它來處理關聯查詢
-->
<select id="getStudents" resultMap="StudentTeacher">
select * from student
</select>
<resultMap id="StudentTeacher" type="Student">
<!--association關聯屬性 property屬性名 javaType屬性類型 column在多的一方的表中的列名-->
<association property="teacher" column="tid" javaType="Teacher" select="getTeacher"/>
</resultMap>
<!--
這裡傳遞過來的id,隻有一個屬性的時候,下面可以寫任何值
association中column多參數配置:
column="{key=value,key=value}"
其實就是鍵值對的形式,key是傳給下個sql的取值名稱,value是片段一中sql查詢的字段名。
-->
<select id="getTeacher" resultType="teacher">
select * from teacher where id = #{id}
</select>
</mapper>
3、編寫完畢去Mybatis配置檔案中,注冊Mapper!
4、注意點說明:
<resultMap id="StudentTeacher" type="Student">
<!--association關聯屬性 property屬性名 javaType屬性類型 column在多的一方的表中的列名-->
<association property="teacher" column="{id=tid,name=tid}" javaType="Teacher" select="getTeacher"/>
</resultMap>
<!--
這裡傳遞過來的id,隻有一個屬性的時候,下面可以寫任何值
association中column多參數配置:
column="{key=value,key=value}"
其實就是鍵值對的形式,key是傳給下個sql的取值名稱,value是片段一中sql查詢的字段名。
-->
<select id="getTeacher" resultType="teacher">
select * from teacher where id = #{id} and name = #{name}
</select>
5、測試
@Test
public void testGetStudents(){
SqlSession session = MybatisUtils.getSession();
StudentMapper mapper = session.getMapper(StudentMapper.class);
List<Student> students = mapper.getStudents();
for (Student student : students){
System.out.println(
"學生名:"+ student.getName()
+"\t老師:"+student.getTeacher().getName());
}
}
按結果嵌套處理
除了上面這種方式,還有其他思路嗎?
我們還可以按照結果進行嵌套處理;
1、接口方法編寫
public List<Student> getStudents2();
2、編寫對應的mapper檔案
<!--
按查詢結果嵌套處理
思路:
1. 直接查詢出結果,進行結果集的映射
-->
<select id="getStudents2" resultMap="StudentTeacher2" >
select s.id sid, s.name sname , t.name tname
from student s,teacher t
where s.tid = t.id
</select>
<resultMap id="StudentTeacher2" type="Student">
<id property="id" column="sid"/>
<result property="name" column="sname"/>
<!--關聯對象property 關聯對象在Student實體類中的屬性-->
<association property="teacher" javaType="Teacher">
<result property="name" column="tname"/>
</association>
</resultMap>
3、去mybatis-config檔案中注入【此處應該處理過了】
4、測試
@Test
public void testGetStudents2(){
SqlSession session = MybatisUtils.getSession();
StudentMapper mapper = session.getMapper(StudentMapper.class);
List<Student> students = mapper.getStudents2();
for (Student student : students){
System.out.println(
"學生名:"+ student.getName()
+"\t老師:"+student.getTeacher().getName());
}
}
小結
按照查詢進行嵌套處理就像SQL中的子查詢
按照結果進行嵌套處理就像SQL中的聯表查詢
一對多的處理
一對多的了解:
- 一個老師擁有多個學生
- 如果對于老師這邊,就是一個一對多的現象,即從一個老師下面擁有一群學生(集合)!
實體類編寫
@Data
public class Student {
private int id;
private String name;
private int tid;
}
@Data
public class Teacher {
private int id;
private String name;
//一個老師多個學生
private List<Student> students;
}
..... 和之前一樣,搭建測試的環境!
1、TeacherMapper接口編寫方法
//擷取指定老師,及老師下的所有學生
public Teacher getTeacher(int id);
2、編寫接口對應的Mapper配置檔案
<mapper namespace="com.kuang.mapper.TeacherMapper">
<!--
思路:
1. 從學生表和老師表中查出學生id,學生姓名,老師姓名
2. 對查詢出來的操作做結果集映射
1. 集合的話,使用collection!
JavaType和ofType都是用來指定對象類型的
JavaType是用來指定pojo中屬性的類型
ofType指定的是映射到list集合屬性中pojo的類型。
-->
<select id="getTeacher" resultMap="TeacherStudent">
select s.id sid, s.name sname , t.name tname, t.id tid
from student s,teacher t
where s.tid = t.id and t.id=#{id}
</select>
<resultMap id="TeacherStudent" type="Teacher">
<result property="name" column="tname"/>
<collection property="students" ofType="Student">
<result property="id" column="sid" />
<result property="name" column="sname" />
<result property="tid" column="tid" />
</collection>
</resultMap>
</mapper>
3、将Mapper檔案注冊到MyBatis-config檔案中
<mappers>
<mapper resource="mapper/TeacherMapper.xml"/>
</mappers>
@Test
public void testGetTeacher(){
SqlSession session = MybatisUtils.getSession();
TeacherMapper mapper = session.getMapper(TeacherMapper.class);
Teacher teacher = mapper.getTeacher(1);
System.out.println(teacher.getName());
System.out.println(teacher.getStudents());
}
public Teacher getTeacher2(int id);
<select id="getTeacher2" resultMap="TeacherStudent2">
select * from teacher where id = #{id}
</select>
<resultMap id="TeacherStudent2" type="Teacher">
<!--column是一對多的外鍵 , 寫的是一的主鍵的列名-->
<collection property="students" javaType="ArrayList" ofType="Student" column="id" select="getStudentByTeacherId"/>
</resultMap>
<select id="getStudentByTeacherId" resultType="Student">
select * from student where tid = #{id}
</select>
@Test
public void testGetTeacher2(){
SqlSession session = MybatisUtils.getSession();
TeacherMapper mapper = session.getMapper(TeacherMapper.class);
Teacher teacher = mapper.getTeacher2(1);
System.out.println(teacher.getName());
System.out.println(teacher.getStudents());
}
1、關聯-association
2、集合-collection
3、是以association是用于一對一和多對一,而collection是用于一對多的關系
4、JavaType和ofType都是用來指定對象類型的
- JavaType是用來指定pojo中屬性的類型
- ofType指定的是映射到list集合屬性中pojo的類型。
注意說明:
1、保證SQL的可讀性,盡量通俗易懂
2、根據實際要求,盡量編寫性能更高的SQL語句
3、注意屬性名和字段不一緻的問題
4、注意一對多和多對一 中:字段和屬性對應的問題
5、盡量使用Log4j,通過日志來檢視自己的錯誤
一對多和多對一對于很多人來說是難點,一定要大量的做練習了解!