mybatisplus怎么多表联查_MySQL两张表联合查询SQL语句

(4) 2024-08-16 17:23

Hi,大家好,我是编程小6,很荣幸遇见你,我把这些年在开发过程中遇到的问题或想法写出来,今天说一说
mybatisplus怎么多表联查_MySQL两张表联合查询SQL语句,希望能够帮助你!!!。

一、前言

最近在加强 ITAEM 团队的一个 app 项目——学生教师学习交流平台
人员组成:安卓 + 前端 + 后台
后台 DAO 层借鉴了华工其他软件开发团队,使用了新颖强大的 MyBatisPlus 框架,里边有一个类似百度贴吧的发帖子的功能:
mybatisplus怎么多表联查_MySQL两张表联合查询SQL语句_https://bianchenghao6.com/blog__第1张
而如果设计表,应为

  • 帖子表 t_post
    - id
    - title 标题
    - content 内容
    - xx
    - user_id 用户外键
  • 用户表 t_user
    + id
    + name 帖子发起者名字
    + xx

示例图中红色框中的内容为 t_user 表的字段 name
而要实现上面显示帖子,就要用到关联查询了,而且帖子很多,必须用分页查询,

那么,怎么通过 MyBatisPlus 来实现关联、分页查询呢 ?很简单,往下看。

二、需求、数据库表设计

这是个人 app 项目中 v1.0 版本的部分表。
mybatisplus怎么多表联查_MySQL两张表联合查询SQL语句_https://bianchenghao6.com/blog__第2张
需求:显示帖子

  • 要帖子基本内容如时间、帖子内容等,即 t_question 表的内容全部要,
  • 同时还要发帖子的人名字,即 t_student 的字段 name

三、代码结构

为了写这篇文章,抽取了该 app 项目中的部分代码,彼此相互关系如下图
mybatisplus怎么多表联查_MySQL两张表联合查询SQL语句_https://bianchenghao6.com/blog__第3张

四、代码实现

1、代码已经放到 github 上了,若对本文的代码有疑问可以去 github 上查看详情:
https://github.com/larger5/MyBatisPlus_page_tables.git

2、entity、mapper、service、controller 使用了 MyBatisPlus 的代码生成器,自动生成大部分基础的代码,操作方法见之前的文章:
在 SpringBoot 中引入 MyBatisPlus 之 常规操作

1.实体

① Question

// import 省略 @TableName("t_question") public class Question implements Serializable { 
    private static final long serialVersionUID = 1L; @ApiModelProperty(value = "问答主键id") @TableId(value = "id", type = IdType.AUTO) private Integer id; @ApiModelProperty(value = "学生外键id") @TableField("student_id") private Integer studentId; @ApiModelProperty(value = "问题内容") private String content; @ApiModelProperty(value = "问题发布时间,发布的时候后台自动生成") private Date date; @ApiModelProperty(value = "问题悬赏的积分") private Integer value; // getter、setter 省略 } 

② Student

// import 省略 @TableName("t_student") public class Student implements Serializable { 
    private static final long serialVersionUID = 1L; @ApiModelProperty(value = "学生主键id") @TableId(value = "id", type = IdType.AUTO) private Integer id; @ApiModelProperty(value = "学生名称") private String name; @ApiModelProperty(value = "学生密码") private String password; @ApiModelProperty(value = "学生积分数") private Integer points; @ApiModelProperty(value = "学生邮件地址") private String email; @ApiModelProperty(value = "学生手机号码") private String phone; @ApiModelProperty(value = "学生学号") private String num; @ApiModelProperty(value = "学生真实姓名") @TableField("true_name") private String trueName; // getter、setter 省略 } 

2.mapper

① StudentMapper

// import 省略 public interface StudentMapper extends BaseMapper<Student> { 
    } 

② QuestionMapper

// import 省略 public interface QuestionMapper extends BaseMapper<Question> { 
    /** * * @param page 翻页对象,可以作为 xml 参数直接使用,传递参数 Page 即自动分页 * @return */ @Select("SELECT t_question.*,t_student.`name` FROM t_question,t_student WHERE t_question.student_id=t_student.id") List<QuestionStudentVO> getQuestionStudent(Pagination page); } 

3、service

① StudentService

// import 省略 public interface StudentService extends IService<Student> { 
    } 

② QuestionService

// import 省略 public interface QuestionService extends IService<Question> { 
    Page<QuestionStudentVO> getQuestionStudent(Page<QuestionStudentVO> page); } 

4、serviceImpl

① StudentServiceImpl

// import 省略 @Service public class StudentServiceImpl extends ServiceImpl<StudentMapper, Student> implements StudentService { 
    } 

② QuestionServiceImpl

// 省略 import @Service public class QuestionServiceImpl extends ServiceImpl<QuestionMapper, Question> implements QuestionService { 
    @Override public Page<QuestionStudentVO> getQuestionStudent(Page<QuestionStudentVO> page) { 
    return page.setRecords(this.baseMapper.getQuestionStudent(page)); } } 

5、controller

// 省略 import @RestController @RequestMapping("/common") @EnableSwagger2 public class CommonController { 
    @Autowired QuestionService questionService; @Autowired StudentService studentService; @GetMapping("/getAllQuestionByPage/{page}/{size}") public Map<String, Object> getAllQuestionByPage(@PathVariable Integer page, @PathVariable Integer size) { 
    Map<String, Object> map = new HashMap<>(); Page<Question> questionPage = questionService.selectPage(new Page<>(page, size)); if (questionPage.getRecords().size() == 0) { 
    map.put("code", 400); } else { 
    map.put("code", 200); map.put("data", questionPage); } return map; } @GetMapping("/getAllQuestionWithStudentByPage/{page}/{size}") public Map<String, Object> getAllQuestionWithStudentByPage(@PathVariable Integer page, @PathVariable Integer size) { 
    Map<String, Object> map = new HashMap<>(); Page<QuestionStudentVO> questionStudent = questionService.getQuestionStudent(new Page<>(page, size)); if (questionStudent.getRecords().size() == 0) { 
    map.put("code", 400); } else { 
    map.put("code", 200); map.put("data", questionStudent); } return map; } } 

6、MyBatisPlus 配置

// 省略 import @EnableTransactionManagement @Configuration @MapperScan("com.cun.app.mapper") public class MybatisPlusConfig { 
    /** * 分页插件 */ @Bean public PaginationInterceptor paginationInterceptor() { 
    return new PaginationInterceptor(); } /** * 打印 sql */ @Bean public PerformanceInterceptor performanceInterceptor() { 
    PerformanceInterceptor performanceInterceptor = new PerformanceInterceptor(); //格式化sql语句 Properties properties = new Properties(); properties.setProperty("format", "true"); performanceInterceptor.setProperties(properties); return performanceInterceptor; } } 

7、关联查询 VO 对象

// import 省略 public class QuestionStudentVO implements Serializable { 
    @ApiModelProperty(value = "问答主键id") @TableId(value = "id", type = IdType.AUTO) private Integer id; @ApiModelProperty(value = "学生外键id") @TableField("student_id") private Integer studentId; private String name; @ApiModelProperty(value = "问题内容") private String content; @ApiModelProperty(value = "问题发布时间,发布的时候后台自动生成") private Date date; @ApiModelProperty(value = "问题悬赏的积分") private Integer value; // getter、setter 省略 

五、测试接口

mybatisplus怎么多表联查_MySQL两张表联合查询SQL语句_https://bianchenghao6.com/blog__第4张

1、没有关联的分页查询接口

http://localhost/common/getAllQuestionByPage/1/2

① json 输出

{ 
    "code": 200, "data": { 
    "total": 10, "size": 2, "current": 1, "records": [ { 
    "id": 1, "studentId": 3, "content": "唐代,渝州城里,有一个性格开朗、乐观的小伙子,名叫景天。", "date": 00, "value": 5 }, { 
    "id": 2, "studentId": 1, "content": "雪见从小父母双亡,由爷爷唐坤抚养成人。", "date": 00, "value": 20 } ], "pages": 5 } } 

② sql 执行

mybatisplus怎么多表联查_MySQL两张表联合查询SQL语句_https://bianchenghao6.com/blog__第5张

2、多表关联、分页查询接口

http://localhost/common/getAllQuestionWithStudentByPage/1/2

① json 输出

{ 
    "code": 200, "data": { 
    "total": 10, "size": 2, "current": 1, "records": [ { 
    "id": 1, "studentId": 3, "name": "vv", "content": "唐代,渝州城里,有一个性格开朗、乐观的小伙子,名叫景天。", "date": 00, "value": 5 }, { 
    "id": 2, "studentId": 1, "name": "cun", "content": "雪见从小父母双亡,由爷爷唐坤抚养成人。", "date": 00, "value": 20 } ], "pages": 5 } } 

② sql 执行

mybatisplus怎么多表联查_MySQL两张表联合查询SQL语句_https://bianchenghao6.com/blog__第6张

六、小结

写本文的原因:

  • ①网上有做法不合时宜的文章(自定义page类、配置版)
  • ②官方文档使用的是配置版的,笔者采用注解版的
MyBatis 配置版 MyBatis 注解版
① 动态 sql 灵活、② xml 格式的 sql,可拓展性好 ① 少一个设置,少一个错误爆发点、② 代码清晰优雅

当然,智者见智仁者见仁

参考资料:
MyBatisPlus 官方文档:分页插件:方式一 、传参区分模式【推荐】

今天的分享到此就结束了,感谢您的阅读,如果确实帮到您,您可以动动手指转发给其他人。

上一篇

已是最后文章

下一篇

已是最新文章

发表回复