mysql - 如何根据三列选择具有 MAX 的行

标签 mysql sql max greatest-n-per-group

我的表定义如下

软件

software_uid  name 
-------------|----------
1            | word 
2            | excel

平台

platform_uid  name 
-------------|----------
1            | windows 
2            | osx

下载

CREATE TABLE IF NOT EXISTS `downloads` (
`id` int(11) NOT NULL,
`software_uid` int(11) NOT NULL,
`platform_uid` int(11) NOT NULL,
`version` decimal(2,1) NOT NULL,
`build` varchar(32) COLLATE utf8_unicode_ci NOT NULL,
`path` varchar(128) COLLATE utf8_unicode_ci NOT NULL,
`date_added` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP
) ENGINE=InnoDB AUTO_INCREMENT=6 DEFAULT CHARSET=utf8 COLLATE=utf8_unicode_ci;

这是下载表中的一些数据:

INSERT INTO `downloads` (`id`, `software_uid`, `platform_uid`, `version`, `build`, `path`, `date_added`) VALUES (1, 1, 1, '0.4', 'Alpha 1', 'downloads/word0.4a1win.zip', '2015-02-06 08:15:09'),
(2, 1, 2, '0.4', 'Alpha 1', 'downloads/word0.4a1osx.zip', '2015-02-06 08:29:47'),
(4, 1, 1, '0.5', 'Beta', 'downloads/word0.5bwin.zip', '2015-02-06 08:29:52'),
(5, 2, 1, '0.1', 'PreAlpha', 'downloads/excel0.1pawin.zip', '2015-02-06 09:14:32');

我想从表中提取两种类型的数据(基本上是两个查询):

  1. 其中将显示适用于所有平台的所有软件的最新版本
  2. 一个用于其他下载,不包括第一个

最后附加软件和平台表中的名称。

我已经开始构建多个级别的子查询,并陷入了一个非常令人讨厌的查询:

SELECT `name`,`path`,`os`,`version`,`build` 
FROM `platform` AS s4 
JOIN 
( 
    SELECT * 
    FROM `software` AS `s2` 
    JOIN 
    ( 
        SELECT 
           `software_uid` AS `guid`, `platform_uid`, MAX(`version`) AS `version`, 
           `path`, `build` 
        FROM `downloads` 
        GROUP BY `platform_uid`, `software_uid` 
    ) AS `s1` ON `s1`.`guid`=`s2`.`software_uid`
) 
AS s3 ON s3.`platform_uid` = s4.`platform_uid`;

它正确地给出了一些字段。 GROUP BY 子句是问题的根源,或者准确地说,执行的第一个子查询没有按预期获取结果,因为它没有正确定义。

我做错了什么?

最佳答案

当您使用group by时,不应有任何既不在group by中也不在聚合函数中的字段。因此,pathbuild 在子查询中是不合适的。相反,您必须通过另一个 join 将它们引入。仅仅因为您执行了 max() 并不意味着其他字段来自同一行。

SELECT `name`,`path`,`os`, d.`version`, d.`build` 
FROM platform p  JOIN 
     software s
     on s.`platform_uid` = p.`platform_uid` JOIN
     (SELECT `software_uid` AS `guid`, `platform_uid`,
             MAX(`version`) AS `version`
      FROM `downloads` 
      GROUP BY `software_uid`, `platform_uid`
     ) dd
     ON s.`guid`= dd.`software_uid` join
     downloads d
     on d.software_uid = dd.guid and d.platform_uid = dd.platfrom_uid and
        d.version = dd.version;

此外,您还存在命名冲突的问题 - name 位于多个表中。您应该始终使用表别名来限定列名。我用表名称的缩写替换了您的别名;这使得更容易遵循查询逻辑。

关于mysql - 如何根据三列选择具有 MAX 的行,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28364932/

相关文章:

mysql - 如何从MySQL数据库中选择每列具有不同条件的两列?

mysql - Sql 查询查找两列中的重复项,其中第一列中的值相同

mysql - 智能 SQL 分组依据

mysql - 在一个数据库列中设置两个索引

java - 如何确定从 Java 文本文件中读取的最大值和最小值

mysql - 如何在 MySQL 中列出给定表的所有索引列

php - 仅来自 Zend_DB Select 语句(子查询)的 COUNT(*)

c# - 让 Linq 与 Mysql 和 Mono 一起玩,这可能吗?

mysql - 如何在mysql中执行查询以获得最大总和?

sql - 如何使用 MAX() 在 SQL 中多次出现 Max 值