简体   繁体   English

LEFT JOIN只是多对一的最新一行

[英]LEFT JOIN of only the latest row from a many-to-one

Been banging my head against the wall and cannot solve this :\\ 一直撞在墙上,无法解决这个问题:

SELECT
  `people`.*,
  `students`.*,
  `student_class_relationships`.*,
  `geo_checkin_on_campus`.`datetime_created` as checkin_time
FROM `student_class_relationships`
  LEFT OUTER JOIN `students`
    ON `student_class_relationships`.`student` = `students`.`id`
  LEFT OUTER JOIN `people`
    ON `students`.`student` = `people`.`id`
  LEFT OUTER JOIN `geo_checkin_on_campus`
    ON `students`.`id` = (
      SELECT MIN(`geo_checkin_on_campus`.`student`)
      FROM `geo_checkin_on_campus`
      WHERE `geo_checkin_on_campus`.`student` = `students`.`id`
    )
 WHERE `class` = 56

The expected result is many rows that have only one entry per students.id . 预期的结果是许多行,每个students.id只有一个条目。

Here is my schema 这是我的架构

It is not the best query from performance perspective, 从性能角度来看,它不是最好的查询,

but just to fix your query here is my attempt: 但只是在这里修复你的查询是我的尝试:

SELECT
  `people`.*,
  `students`.*,
  `student_class_relationships`.*,
  geoCheckinOnCampus.datetimeCreated as checkin_time
FROM `student_class_relationships`
  LEFT JOIN `students`
    ON `student_class_relationships`.`student` = `students`.`id`
  LEFT JOIN `people`
    ON `students`.`student` = `people`.`id`
  LEFT JOIN 
    (
      SELECT 
        student,
        MAX(datetime_created) datetimeCreated
      FROM `geo_checkin_on_campus`
      GROUP BY `student`
    ) geoCheckinOnCampus
  ON `students`.`id` = geoCheckinOnCampus.`student`
 WHERE `class` = 56

Note According to @xQbert answer I would really change MIN to MAX function if you are looking for the latest datetime. 注意根据@xQbert的答案,如果你正在寻找最新的日期时间我真的会改变MINMAX功能。

If i assume you want the most recent checkin (and not the earliest created date) for each student in go_checkin_on_Campus then... 如果我假设你想要go_checkin_on_Campus中每个学生的最新签到(而不是最早创建的日期)那么......

SELECT
  `people`.*,
  `students`.*,
  `student_class_relationships`.*,
  B.`datetime_Updated` as checkin_time
FROM `student_class_relationships`
  LEFT OUTER JOIN `students`
    ON `student_class_relationships`.`student` = `students`.`id`
  LEFT OUTER JOIN `people`
    ON `students`.`student` = `people`.`id`
  LEFT OUTER JOIN (
      SELECT max(datetime_updated), student
      FROM `geo_checkin_on_campus`
      group by student
) B
    ON `students`.`id` = B.Student
 WHERE `class` = 56

NOTE: This is a probable answer. 注意:这是一个可能的答案。 I will edit / modify this according to comments from OP. 我将根据OP的评论编辑/修改它。

This basically does nothing. 这基本上什么也没做。 This is as good as just joining on the student id 这和加入学生ID一样好

LEFT OUTER JOIN `geo_checkin_on_campus`
    ON `students`.`id` = (
      SELECT MIN(`geo_checkin_on_campus`.`student`)
      FROM `geo_checkin_on_campus`
      WHERE `geo_checkin_on_campus`.`student` = `students`.`id`
    )

If you want the min (or earliest) datetime_created use something like 如果你想要min(或最早)datetime_created使用类似的东西

LEFT OUTER JOIN (
      SELECT `geo_checkin_on_campus`.`student` student,
              MIN(`geo_checkin_on_campus`.`datetime_created`) dt
      FROM   `geo_checkin_on_campus`
      WHERE  `geo_checkin_on_campus`.`student` = `students`.`id`
      GROUP BY `geo_checkin_on_campus`.`student`
    ) t
    ON `students`.`id` = t.student

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM