mysql - MariaDB 上的 "GROUP BY"与 MySQL 的行为不同

标签 mysql database group-by sql-order-by mariadb

我多次被告知相同的查询 MariaDB 将像在 MySQL 上一样工作......直到我遇到这个问题。

最近,我试图将一个应用程序从MySQL(InnoDB) 克隆到MariaDB(XtraDB)。 尽管 MariaDB 无需更改任何内容即可运行 MySQL 查询,但我惊讶地发现相同的查询在两个平台上的行为实际上完全不同,尤其是在 ORDER BYGROUP BY 中。

举个例子:

    MyTable
    =======
    +----+----------+---------------------+-----------+
    | id | parentId | creationDate        | name      |
    +----+----------+---------------------+-----------+
    | 1  | 2357     | 2017-01-01 06:03:40 | Anna      |
    +----+----------+---------------------+-----------+
    | 2  | 5480     | 2017-01-02 07:13:20 | Becky     |
    +----+----------+---------------------+-----------+
    | 3  | 2357     | 2017-01-03 08:20:12 | Christina |
    +----+----------+---------------------+-----------+
    | 4  | 2357     | 2017-01-03 08:20:15 | Dorothy   |
    +----+----------+---------------------+-----------+
    | 5  | 5480     | 2017-01-04 09:25:45 | Emma      |
    +----+----------+---------------------+-----------+
    | 6  | 1168     | 2017-01-05 10:30:10 | Fiona     |
    +----+----------+---------------------+-----------+
    | 7  | 5480     | 2017-01-05 10:33:23 | Gigi      |
    +----+----------+---------------------+-----------+
    | 8  | 1168     | 2017-01-06 12:46:34 | Heidi     |
    +----+----------+---------------------+-----------+
    | 9  | 1168     | 2017-01-06 12:46:34 | Irene     |
    +----+----------+---------------------+-----------+
    | 10 | 2357     | 2017-01-07 14:58:37 | Jane      |
    +----+----------+---------------------+-----------+
    | 11 | 2357     | 2017-01-07 14:58:37 | Katy      |
    +----+----------+---------------------+-----------+

基本上,我想从查询中获得的是来自每个 GROUPing 的最新记录(即 parentId)。最新的,我的意思是 MAX(creationDate) 和 MAX(id)

所以,对于上面的例子,由于只有三个不同的parentId值,我希望得到:

    +----+----------+---------------------+-----------+
    | id | parentId | creationDate        | name      |
    +----+----------+---------------------+-----------+
    | 11 | 2357     | 2017-01-07 14:58:37 | Katy      |
    +----+----------+---------------------+-----------+
    | 9  | 1168     | 2017-01-06 12:46:34 | Irene     |
    +----+----------+---------------------+-----------+
    | 7  | 5480     | 2017-01-05 10:33:23 | Gigi      |
    +----+----------+---------------------+-----------+

最初应用程序有类似这种方式的查询:

SELECT * FROM
  ( SELECT * FROM `MyTable` WHERE `parentId` IN (...)
    ORDER BY `creationDate` DESC, `id` DESC ) AS `t` 
  GROUP BY `parentId`;

MySQL 上,这是有效的,因为内部查询将排序,然后外部查询从内部查询的结果中获取每个 GROUP 的第一个。外部查询基本上服从内部查询的顺序。

但在 MariaDB 上,外部查询将忽略内部查询结果的排序。我在 MariaDB 上得到这个:

    +----+----------+---------------------+-----------+
    | id | parentId | creationDate        | name      |
    +----+----------+---------------------+-----------+
    | 1  | 2357     | 2017-01-01 06:03:40 | Anna      |
    +----+----------+---------------------+-----------+
    | 2  | 5480     | 2017-01-02 07:13:20 | Becky     |
    +----+----------+---------------------+-----------+
    | 6  | 1168     | 2017-01-05 10:30:10 | Fiona     |
    +----+----------+---------------------+-----------+

为了在 MariaDB 上实现相同的行为,我提出了类似的方法。 (虽然不确定这是否准确。)

SELECT `t1`.* FROM `MyTable` `t1` LEFT JOIN `MyTable` `t2` ON (
        `t1`.`parentId` = `t2`.`parentId`
    AND `t2`.`parentId` IN (...)
    AND `t1`.`creationDate` <= `t2`.`creationDate`
    AND `t1`.`id` < `t2`.`id`)
  ) WHERE `t2`.`id` IS NULL;

现在的问题是……如果我要重写查询,我必须重写数百个……它们之间有些许不同。

我想知道这里是否有人有任何想法可以让我做出尽可能少的改变。

提前谢谢大家。

最佳答案

是的,这是一个仅链接的答案。但链接指向 MariaDB 站点。

这里是另一个关于“不兼容”的讨论:https://mariadb.com/kb/en/mariadb/group-by-trick-has-been-optimized-away/

从技术上讲,MySQL 实现了对 Ansi 标准的扩展。很久以后,它决定删除它,所以我想你会发现 MySQL 已经迁移到 MariaDB。

这里是进行分组最大化的“快速”方法列表,这可能是您正在尝试做的:https://mariadb.com/kb/en/mariadb/groupwise-max-in-mariadb/

关于mysql - MariaDB 上的 "GROUP BY"与 MySQL 的行为不同,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42312547/

相关文章:

python - 加密将捆绑在 pyexe 文件中的 Sqlite 数据库文件

sql - 转储所有模式和触发器并将它们导入另一个 Oracle 数据库

database - 将联系表格 7 数据保存到自定义数据库而不是 wordpress 数据库中

MYSQL:如何使用 GROUP_CONCAT 和分隔符使用 groupby 附加空值?

mysql - 如何摆脱 HAVING 子句中的非分组字段

mysql - 存储过程中的多案例选择

mysql - 使用 mysql 查询跟踪 api 调用

php - php错误中的html表

php - LOAD DATA INFILE + 进度跟踪

mysql - 一起使用 ORDER BY 和 GROUP BY