【发布时间】:2017-02-18 07:42:17
【问题描述】:
我多次被告知,相同的查询 MariaDB 将像在 MySQL 上一样工作......直到我遇到这个问题。
最近,我正在尝试将应用程序从 MySQL(InnoDB) 克隆到 MariaDB(XtraDB)。 尽管 MariaDB 无需更改任何内容即可运行 MySQL 查询,但我惊讶地发现,相同的查询实际上在两个平台上的行为完全不同,尤其是在 ORDER BY 和 GROUP 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;
现在的问题是……如果我要重写查询,我必须重写数百个……而且它们彼此之间有些不同。
我想知道这里是否有人有任何想法可以让我做出尽可能少的更改。
提前谢谢大家。
【问题讨论】:
-
MySQL 手册涵盖了这一点。它提供了 3 个有效的解决方案。如果你还在苦苦挣扎,请参阅meta.stackoverflow.com/questions/333952/…
-
检查你的mysql和mariaDB版本并检查select中的列必须由group by子句使用的方式..你的问题是否与你的db版本中的only_full_group_by设置模式不同的行为有关..你可以看到一些信息dev.mysql.com/doc/refman/5.7/en/sql-mode.html
-
MySQL 版本是 5.5.52... MariaDB 版本是 10.1.21... 我尝试在 MariaDB 上运行 SET sql_mode = "ONLY_FULL_GROUP_BY" 但没有区别。
标签: mysql database group-by sql-order-by mariadb