Java企业级开发学习笔记(2.3)利用MyBatis实现关联查询

news2024/11/18 16:52:21

该文章主要为完成实训任务,详细实现过程及结果见【http://t.csdn.cn/ZVEZd】

文章目录

  • 一、创建数据库表
    • 1.1 创建教师表
    • 1.2 创建班级表
    • 1.3 创建学生表
  • 二、创建于数据库表对应的实体类
    • 2.1 创建教师实体类
    • 2.2 创建学生实体类
    • 2.3 创建班级实体类
  • 三、创建班级映射器配置文件
  • 四、修改MyBatis配置文件
  • 五、创建班级映射接口
  • 六、创建班级映射器测试类
    • 6.1 测试按编号查询班级方法
    • 6.2 测试查询全部班级方法
  • 七、课后作业
    • 7.1 创建三张表:学生表、选课表、课程表
    • 7.2 给三张表输入若干记录
    • 7.3 查询任务
      • 7.3.1 按学号查询学生选课及成绩
      • 7.3.2 查询全部学生选课及成绩


一、创建数据库表

1.1 创建教师表

  • 执行SQL语句,创建教师表t_teacher
CREATE TABLE `t_teacher` (
`t_id` int(11) NOT NULL AUTO_INCREMENT,
`t_name` varchar(20) DEFAULT NULL,
PRIMARY KEY (`t_id`)
) ENGINE=InnoDB AUTO_INCREMENT=4 DEFAULT CHARSET=utf8mb4;

在这里插入图片描述

  • 执行SQL语句,插入3条记录
INSERT INTO `t_teacher` VALUES ('1', '刘晓云');
INSERT INTO `t_teacher` VALUES ('2', '郑同华');
INSERT INTO `t_teacher` VALUES ('3', '李明军');

在这里插入图片描述

  • 查看教师表记录
    在这里插入图片描述

1.2 创建班级表

  • 执行SQL语句,创建班级表t_class
CREATE TABLE `t_class` (
  `c_id` int(11) NOT NULL AUTO_INCREMENT,
  `c_name` varchar(20) DEFAULT NULL,
  `teacher_id` int(11) DEFAULT NULL,
PRIMARY KEY (`c_id`),
KEY `fk_teacher_id` (`teacher_id`),
CONSTRAINT `fk_teacher_id` FOREIGN KEY (`teacher_id`) REFERENCES `t_teacher` (`t_id`)
) ENGINE=InnoDB AUTO_INCREMENT=4 DEFAULT CHARSET=utf8mb4;

在这里插入图片描述

  • 执行SQL语句,插入3条记录
INSERT INTO `t_class` VALUES ('1', '2021软件1班', '3');
INSERT INTO `t_class` VALUES ('2', '2021软件2班', '2');
INSERT INTO `t_class` VALUES ('3', '2021软件3班', '1');

在这里插入图片描述

  • 查看班级表记录
    在这里插入图片描述

1.3 创建学生表

  • 执行SQL语句,创建学生表t_student
CREATE TABLE `t_student` (
  `s_id` int(11) NOT NULL AUTO_INCREMENT,
  `s_name` varchar(30) DEFAULT NULL,
  `s_gender` varchar(10) DEFAULT NULL,
  `s_age` int(11) DEFAULT NULL,
  `class_id` int(11) DEFAULT NULL,
  PRIMARY KEY (`s_id`)
) ENGINE=InnoDB AUTO_INCREMENT=10 DEFAULT CHARSET=utf8mb4;

在这里插入图片描述

  • 执行SQL语句,插入9条记录
INSERT INTO `t_student` VALUES ('1', '佟大为', '男', '20', '1');
INSERT INTO `t_student` VALUEs ('2', '李方玉', '女', '19', '1');
INSERT INTO `t_student` VALUES ('3', '郑大林', '男', '19', '2');
INSERT INTO `t_student` VALUES ('4', '温晓辉', '男', '18', '1');
INSERT INTO `t_student` VALUES ('5', '吴文静', '女', '19', '2');
INSERT INTO `t_student` VALUES ('6', '肖文艳', '女', '18', '3');
INSERT INTO `t_student` VALUES ('7', '杨文军', '男', '20', '3');
INSERT INTO `t_student` VALUES ('8', '唐雨涵', '女', '19', '2');
INSERT INTo `t_student` VALUES ('9', '金雨欣', '女', '20', '2');

在这里插入图片描述

  • 查看学生表记录
    在这里插入图片描述

二、创建于数据库表对应的实体类

2.1 创建教师实体类

  • cn.kox.mybatis.bean包里创建Teacher
    在这里插入图片描述
package cn.kox.mybatis.bean;

/**
 * @ClassName: Teacher
 * @Author: Kox
 * @Data: 2023/4/12
 * @Sketch:
 */
public class Teacher {
    private int id;
    private String name;

    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;
    }

    @Override
    public String toString() {
        return "Teacher{" +
                "id=" + id +
                ", name='" + name + '\'' +
                '}';
    }
}

2.2 创建学生实体类

  • cn.kox.mybatis.bean包里创建Student
    在这里插入图片描述
package cn.kox.mybatis.bean;

/**
 * @ClassName: Student
 * @Author: Kox
 * @Data: 2023/4/12
 * @Sketch:
 */
public class Student {
    private int id;
    private String name;
    private String gender;
    private int age;
    private Clazz clazz;

    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 int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    public Clazz getClazz() {
        return clazz;
    }

    public void setClazz(Clazz clazz) {
        this.clazz = clazz;
    }

    @Override
    public String toString() {
        return "Student{" +
                "id=" + id +
                ", name='" + name + '\'' +
                ", gender='" + gender + '\'' +
                ", age=" + age +
                ", clazz=" + clazz.getName() +
                '}';
    }
}

2.3 创建班级实体类

  • cn.kox.mybatis.bean包里创建Clazz
    在这里插入图片描述
package cn.kox.mybatis.bean;

import java.util.List;

/**
 * @ClassName: Clazz
 * @Author: Kox
 * @Data: 2023/4/12
 * @Sketch:
 */
public class Clazz {
    private int id;
    private String name;
    private Teacher teacher;
    private List<Student> students;

    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 Teacher getTeacher() {
        return teacher;
    }

    public void setTeacher(Teacher teacher) {
        this.teacher = teacher;
    }

    public List<Student> getStudents() {
        return students;
    }

    public void setStudents(List<Student> students) {
        this.students = students;
    }

    @Override
    public String toString() {
        return "Clazz{" +
                "id=" + id +
                ", name='" + name + '\'' +
                ", teacher=" + teacher.getName() +
                ", students=" + students +
                '}';
    }
}

三、创建班级映射器配置文件

  • resources/mapper目录里创建班级映射器配置文件ClazzMapper.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="cn.kox.mybatis.mapper.ClazzMapper">
    <!--按编号查询班级(带教师信息和学生信息),需要三表关联查询-->
    <select id="findById" parameterType="int" resultMap="clazzMap1">
        SELECT * FROM t_class
                          INNER JOIN t_teacher ON t_class.teacher_id = t_teacher.t_id
                          INNER JOIN t_student ON t_class.c_id = t_student.class_id
        WHERE c_id = #{id};
    </select>

    <!--定义班级结果映射-->
    <resultMap id="clazzMap1" type="Clazz">
        <result property="id" column="c_id"/>
        <result property="name" column="c_name"/>
        <!--一对一关联:一个班级配一个老师-->
        <association property="teacher" javaType="Teacher" column="teacher_id">
            <result property="id" column="t_id"/>
            <result property="name" column="t_name"/>
        </association>
        <!--一对多关联:一个班级有多个学生-->
        <collection property="students" ofType="Student">
            <result property="id" column="s_id"/>
            <result property="name" column="s_name"/>
            <result property="gender" column="s_gender"/>
            <result property="age" column="s_age"/>
            <association property="clazz" javaType="Clazz" column="class_id">
                <result property="name" column="c_name"/>
            </association>
        </collection>
    </resultMap>

    <!--##########################################################-->

    <!--查询全部班级(带教师信息,不带学生信息),利用嵌套查询实现一对一关联-->
    <select id="findAll" resultMap="clazzMap2">
        SELECT * FROM t_class;
    </select>

    <!--定义班级结果映射-->
    <resultMap id="clazzMap2" type="Clazz">
        <result property="id" column="c_id"/>
        <result property="name" column="c_name"/>
        <!--通过select属性定义子查询-->
        <association property="teacher" javaType="Teacher" column="teacher_id" select="getTeacher"/>
    </resultMap>

    <!--按编号查询教师-->
    <select id="getTeacher" resultType="Teacher">
        SELECT t_id id, t_name name FROM t_teacher WHERE t_id = #{id};
    </select>
</mapper>

四、修改MyBatis配置文件

  • 在配置文件里配置班级、教师与学生实体类别名,以及班级映射器配置文件
    在这里插入图片描述

五、创建班级映射接口

  • cn.kox.mybatis.mapper包里创建ClazzMapper接口
    在这里插入图片描述
package cn.kox.mybatis.mapper;

import cn.kox.mybatis.bean.Clazz;

import java.util.List;

public interface ClazzMapper {
    Clazz findById(int id); // 按编号查询班级记录
    List<Clazz> findAll();   // 查询全部班级记录
}

六、创建班级映射器测试类

  • test/java目录的cn.kox.mybatis.mapper包里创建TestClazzMapper
    在这里插入图片描述
package cn.kox.mybatis.mapper;

import cn.kox.mybatis.bean.Clazz;
import cn.kox.mybatis.bean.Student;
import org.apache.ibatis.io.Resources;
import org.apache.ibatis.session.SqlSession;
import org.apache.ibatis.session.SqlSessionFactory;
import org.apache.ibatis.session.SqlSessionFactoryBuilder;
import org.junit.After;
import org.junit.Before;
import org.junit.Test;

import java.io.IOException;
import java.io.Reader;
import java.util.List;

/**
 * @ClassName: TestClazzMapper
 * @Author: Kox
 * @Data: 2023/4/12
 * @Sketch:
 */
public class TestClazzMapper {

    private SqlSession sqlSession; // SQL会话
    private ClazzMapper clazzMapper; // 班级映射器

    @Before
    public void init() {
        try {
            // 读取MyBatis配置文件
            Reader reader = Resources.getResourceAsReader("mybatis-config.xml");
            // 基于MyBatis配置文件构建SQL会话工厂
            SqlSessionFactory factory = new SqlSessionFactoryBuilder().build(reader);
            // 利用SQL会话工厂获取SQL会话
            sqlSession = factory.openSession();
            // 利用SQL会话获取班级映射器对象
            clazzMapper = sqlSession.getMapper(ClazzMapper.class);
            // 提示用户SQL会话创建成功
            System.out.println("SQL会话创建成功~");
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    @Test // 按编号查询班级(带教师信息和学生信息)
    public void testFindById() {
        // 查询id为1的班级信息
        int id = 1;
        Clazz clazz = clazzMapper.findById(id);
        // 判断是否查询成功
        if (clazz != null) {
            // 打印班级信息
            System.out.println(clazz);
            // 打印班级学生信息
            System.out.println("班级编号为[" + id + "]的学生:");
            List<Student> students = clazz.getStudents();
            students.forEach(student -> System.out.println(student));

        } else {
            System.out.println("编号为[" + id + "]的班级不存在~");
        }
    }

    @Test // 查询全部班级信息
    public void testFindAll() {
        // 查询全部班级
        List<Clazz> clazzes = clazzMapper.findAll();
        // 输出全部班级
        clazzes.forEach(clazz -> System.out.println(clazz));
    }

    @After
    public void destroy() {
        // 关闭SQL会话
        sqlSession.close();
        // 提示用户SQL会话关闭
        System.out.println("SQL会话已经关闭~");
    }
}

6.1 测试按编号查询班级方法

  • 运行测试方法testFindById(),查看结果
    在这里插入图片描述

6.2 测试查询全部班级方法

在这里插入图片描述

七、课后作业

7.1 创建三张表:学生表、选课表、课程表

  • 学生表(学号, 姓名, 性别, 年龄, 电话) - 学号是字符串类型
  • 选课表(学号, 课程号, 成绩)
  • 课程表(课程号, 课程名, 课时数) - 课程号是字符串类型

7.2 给三张表输入若干记录

7.3 查询任务

7.3.1 按学号查询学生选课及成绩

  • 查询结果包含字段:学号、姓名、课程名、成绩

7.3.2 查询全部学生选课及成绩

  • 查询结果包含字段:学号、姓名、课程名、成绩

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/419312.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Leetcode394 字符串解码 递归和非递归

字符串解码 https://leetcode.cn/problems/decode-string/ 给定一个经过编码的字符串&#xff0c;返回它解码后的字符串。 编码规则为: k[encoded_string]&#xff0c;表示其中方括号内部的 encoded_string 正好重复 k 次。注意 k 保证为正整数。 你可以认为输入字符串总是有效…

你搞清楚了吗?| GET请求方式的长度限制到底是多少?

目录 &#x1f4cd; 浏览器限制 &#x1f4cd; 服务器限制 在大多数人的一贯认识中&#xff0c;一直认为get请求方式有2048B的长度限制&#xff0c;其实这种说法是有失偏颇的&#xff0c;甚至可以说是错误的。 这个问题一直以来似乎是被N多人误解&#xff0c;其实Http Get方…

2.3-3单链表的查找

按位查找&#xff1a; so easy (1)边界情况 i0 没有循环&#xff0c;直接返回头节点 &#xff08;2&#xff09;如果i8 当不合法&#xff0c;返回NULL. (3)普通情况i3;(平均时间复杂度为O&#xff08;n&#xff09;) 进行封装&#xff1a;避免重复&#xff0c;更加简洁更…

CRYSTALS-Dilithium

文章目录简介1和2版本区别2和3的区别1.介绍1.1基本方法概述密钥生成算法签名过程验证1.2Dilithium实现注意事项安全性。基础操作环操作模约简。元素的大小NTT域表示2.3 HashingsignatureGenζ ← {0,1}256\{{0, 1\}}^{256}{0,1}256(ρ, ρ, K) ∈ {0,1}256\{{0, 1\}}^{256}{0,1…

【springBoot篇2】springBoot日志篇

目录 一、日志有什么作用 作用1&#xff1a;快速定位问题的所在之处(最主要) 作用2&#xff1a;记录用户的登录日志 作用3&#xff1a;记录系统的操作日志 作用4&#xff1a;记录方法的执行时间 二、日志怎样使用 ①先得到日志对象(slf4j的Logger对象) ​​​编辑 ②根…

Python 无监督学习实用指南:1~5

原文&#xff1a;Hands-on unsupervised learning with Python 协议&#xff1a;CC BY-NC-SA 4.0 译者&#xff1a;飞龙 本文来自【ApacheCN 深度学习 译文集】&#xff0c;采用译后编辑&#xff08;MTPE&#xff09;流程来尽可能提升效率。 不要担心自己的形象&#xff0c;只关…

docker运行服务端性能监控系统Prometheus和数据分析系统Grafana

文章目录一、Prometheus的安装和运行1、使用docker拉取镜像2、创建prometheus.yml文件3、启动容器4、查看启动是否成功5、记录安装过程中出现的错误二、Grafana的安装和运行1、使用docker拉取镜像2、创建grafana3、运行grafana4、查看grafana运行日志5、登录grafana一、Prometh…

学会了selenium 模拟鼠标操作,你就可以偷懒点点点了

目录&#xff1a;导读 前言 01.ActionChains 类常用方法 02.ActionChains 类所有方法 03.ActionChains 使用步骤 04.实战 05.总结 前言 我们在做 Web 自动化的时候&#xff0c;有时候页面的元素不需要我们点击&#xff0c;值需要把鼠标移动上去就能展示各种信息。 这个…

云服务器开启声音的办法

云服务器开启声音的办法 淘小云 云计算从业者&#xff0c;架构师 ​关注 云服务器没有声音是一件很烦恼的事情&#xff0c;那么今天小编给大家分享一下服务器开启声音的方法&#xff01; 首先您需要先拥有一台windows系统的云服务器&#xff0c;小编这里以腾讯云服务器为例…

( “树” 之 DFS) 687. 最长同值路径 ——【Leetcode每日一题】

687. 最长同值路径 给定一个二叉树的 root &#xff0c;返回 最长的路径的长度 &#xff0c;这个路径中的 每个节点具有相同值 。 这条路径可以经过也可以不经过根节点。 两个节点之间的路径长度 由它们之间的边数表示。 示例 1: 输入&#xff1a;root [5,4,5,1,1,5] 输出&…

Loki采集Mysql errorlog,你值得拥有的错误日志聚合系统

说到分布式日志存储系统&#xff0c;大家肯定对ELK、EFK这些工具并不陌生。可是它们都基于Elasticsearch存储&#xff0c;搭建复杂&#xff0c;耗资源&#xff0c;上手难。所以&#xff0c;个人非常推崇Grafana Labs开源的Loki 轻量级日志聚合分析系统 Loki使用标签来作为索引…

java maven学习

我本身是个前端&#xff0c;目前Java都算是自学&#xff0c;为了直接能做Java服务端开发&#xff0c;我其实很多基础都没学&#xff0c;直接上项目做东西。现在来补充常用maven。避免以后出现低级错误。 一、依赖拉不下来解决步骤 1. 检查网络&#xff08;最好检查&#xff09…

Flutter(六)可滚动组件

目录1.可滚动组件简介Sliver布局模型ScrollableViewportSliver可滚动组件的通用配置2.SingleChildScrollView3.ListView默认构造函数ListView.builderListView.separated固定高度列表ListView 原理无限加载列表&#xff0c;分页添加Header4.滚动监听及控制ScrollController滚动…

SpringCloud-Alibaba学习笔记01——Nacos介绍以及注册中心的演变和Nacos核心功能

文章参考自图灵大佬课程&#xff1a;https://www.bilibili.com/video/BV1fe4y1b7ha?p1&vd_source5f425e0074a7f92921f53ab87712357b 1.什么是Nacos 官方&#xff1a;一个更易于构建云原生应用的动态服务发现(Nacos Discovery )、服务配置(Nacos Config)和服务管理平台。 …

threejs 模型 世界坐标系和设备坐标系

前言 开发中遇到需求需要点击屏幕位置处&#xff0c;生成一个类似圆形弹窗面板&#xff0c;这个交互需要进行的坐标转换为模型坐标&#xff08;局部坐标&#xff09;>场景坐标&#xff08;世界坐标&#xff09;>标准设备坐标>屏幕空间坐标&#xff0c;也就是一个将3D…

开启分片支持需要如何去做?

开启分片支持 如果您计划使您的Javashop系统数据分片&#xff0c;请参考本文档进行相应的配置。 一、做好分片策略 在开始之前&#xff0c;您应该根据自己的业务情况准备好分片策略&#xff0c;包括&#xff1a; 1、要用几个数据库来分片 2、相应的表要分几张表 在本例&#x…

大数据 | 实验一:大数据系统基本实验 | MapReduce 初级编程

文章目录&#x1f4da;实验目的&#x1f4da;实验平台&#x1f4da;实验内容&#x1f407;编程实现文件的合并和去重&#x1f407;编程实现对输入文件的排序&#x1f407;对指定的表格进行信息挖掘&#x1f4da;实验目的 1&#xff09;通过实验掌握基本的 MapReduce 编程方法。…

警惕“Money Message”勒索软件!数据安全不容忽视

近段时间&#xff0c;出现了一个名为“Money Message”的新型勒索软件&#xff0c;他们利用Money Message 病毒加密文件并以此向受害者勒索巨额赎金。 Money Message勒索软件是用 C编写&#xff0c;包含一个嵌入式JSON 配置文件&#xff0c;用于确定设备的加密方式。加密设备后…

小红书内容种草,曝光渠道分析总结

这是一个内容为王的时代&#xff0c;也是一个内容爆炸的时代。想要在以分享特色的小红书平台&#xff0c;实现内容种草&#xff0c;迅速出圈。今天来马文化传媒就从实操的角度&#xff0c;为大家带来小红书内容种草&#xff0c;曝光渠道分析总结的各种干货&#xff01; 一、什…

关于图形界面Pyqt与QT的区别选择

关于图像界面&#xff08;GUI&#xff09;想必大家都并不陌生&#xff0c;想要将一段已经完善的功能列表进行可视化操作并且具有一定的操作空间&#xff0c;将功能可视化必不可少&#xff0c;一个好的可视化工具不仅可以集成一系列小的文件功能&#xff0c;还能将不同方法之间的…