MySql5.7及以上 ORDER BY 報錯問題

疯子丶pony發表於2024-10-25

一、問題

本人使用的MySql版本是8.0的

當MySql5.7及以上的版本執行帶有 ORDER BY 的SQL語句時可能會報錯。

例如,執行以下mysql語句:

SELECT id, user_id, title FROM m_article WHERE user_id>=100 AND user_id <=200 GROUP BY user_id;

SQL報錯資訊如下:

1055 - Expression #1 of SELECT list is not in GROUP BY clause and contains nonaggregated column 'test.m_article.id' which is not functionally dependent on columns in GROUP BY clause; this is incompatible with sql_mode=only_full_group_by

二、分析原因

SQL-92及更早版本的查詢不允許使用select列表、HAVING條件或ORDER BY列表引用未在GROUP BY子句中命名的非聚合列。

簡單來說:由於sql-mode的引數配置了ONLY_FULL_GROUP_BY,這時 select 的欄位不在 group by 中,並且 select 的欄位未使用聚合函式(SUM,MAX,MIN等)的話,那麼這條SQL查詢是被 MySql 認為非法。

MySql官方文件:https://dev.mysql.com/doc/refman/8.0/en/group-by-handling.html

三、解決方法

1、不修改 sql-mode 的引數情況下

1.1、以本文中的例句修改,在原來的 ORDER BY 後面多加一個主鍵。

1.2、以本文中的例句修改,使用ANY_VALUE()函式,把非 GROUP BY 列中的欄位和沒有使用聚合函式的都加上。使用ANY_VALUE()不檢查函式結果是否為ONLY_FULL_GROUP_BY SQL模式。

MySql官方文件:https://dev.mysql.com/doc/refman/8.0/en/miscellaneous-functions.html#function_any-value

2、修改 sql-mode 的引數

2.2、使用SQL語句臨時修改。

SET sql_mode ='STRICT_TRANS_TABLES,NO_ZERO_IN_DATE,NO_ZERO_DATE,ERROR_FOR_DIVISION_BY_ZERO,NO_ENGINE_SUBSTITUTION';

2.3、修改 MySql 的配置檔案,這種方式配置完成後都要重啟MySql。

2.3.1、Linux 中找到 MySql 配置檔案,檔名一般叫【my.cnf】,檔案路徑一般在:/etc/my.cnf,/etc/mysql/my.cnf。開啟【my.cnf】檔案後,就在 [mysqld] 下面追加一行即可。

sql-mode=STRICT_TRANS_TABLES,NO_ZERO_IN_DATE,NO_ZERO_DATE,ERROR_FOR_DIVISION_BY_ZERO,NO_AUTO_CREATE_USER,NO_ENGINE_SUBSTITUTION

2.3.2、Windows 中找到【my.ini】檔案,開啟後在 [mysqld] 下面追加一行即可。

sql-mode=STRICT_TRANS_TABLES,NO_ZERO_IN_DATE,NO_ZERO_DATE,ERROR_FOR_DIVISION_BY_ZERO,NO_AUTO_CREATE_USER,NO_ENGINE_SUBSTITUTION

相關文章