msyql sql mode 问题

博客围绕MySQL查询报错“ORDER BY clause is not in GROUP BY..this is incompatible with sql_mode=only_full_group_by”展开,介绍了两种解决方案,一是修改mysql配置文件,但可能导致数据丢失;二是不修改配置文件,修改sql写法。还对sql_mode进行了解析。

mysql查询报错: ORDER BY clause is not in GROUP BY..this is incompatible with sql_mode=only_full_group_by

2017年07月25日 10:34:30 tyilack_小小黑 阅读数:42333

 版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.youkuaiyun.com/u014520745/article/details/76056170

首先谢谢评论区大神指出错误,旧方法修改配置文件缺失存在问题,下面给出两种解决方案:

一、旧方法,修改mysql配置文件,但是会导致数据丢失等不可预知的错误

在用mysql执行如下查询的时候:

select * from `sys_user_group` group by `GROUP_ID`
  • 1

报错信息如下:

[Err] 1055 - Expression #1 of ORDER BY clause is not in GROUP BY clause and contains nonaggregated column 'information_schema.PROFILING.SEQ' which is not functionally dependent on columns in GROUP BY clause; this is incompatible with sql_mode=only_full_group_by
  • 1

错误的原因是我mysql版本是5.7的,使用如下语句查询可知

select @@sql_mode;
  • 1

这里写图片描述

里面默认设置了

sql_mode=only_full_group_by
  • 1

only_full_group_by:使用这个就是使用和oracle一样的group 规则, select的列都要在group中,或者本身是聚合列(SUM,AVG,MAX,MIN) 才行,其实这个配置目前个人感觉和distinct差不多的,所以去掉就好

直接修改mysql配置文件(我的系统是Ubuntu16.04的,在/etc/mysql/mysql.conf.d/mysqld.cnf中并没有sql_mode这个配置,所以直接加上就好,如果是其他系统有得修改就不用添加了)

[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
  • 1

到此mysql配置文件的完整配置如下:

[mysqld_safe]
socket          = /var/run/mysqld/mysqld.sock
nice            = 0

[mysqld]
#
# * Basic Settings
#
user            = mysql
pid-file        = /var/run/mysqld/mysqld.pid
socket          = /var/run/mysqld/mysqld.sock
port            = 3306
basedir         = /usr
datadir         = /var/lib/mysql
tmpdir          = /tmp
lc-messages-dir = /usr/share/mysql
skip-external-locking
lower_case_table_names=1
character-set-server=utf8
sql_mode=STRICT_TRANS_TABLES,NO_ZERO_IN_DATE,NO_ZERO_DATE,ERROR_FOR_DIVISION_BY_ZERO,NO_AUTO_CREATE_USER,NO_ENGINE_SUBSTITUTION
#
# Instead of skip-networking the default is now to listen only on
# localhost which is more compatible and is not less secure.
#bind-address           = 127.0.0.1
#
# * Fine Tuning
#
key_buffer_size         = 16M
max_allowed_packet      = 16M
thread_stack            = 192K

  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31

二、不修改配置文件,修改sql写法

在分组时,5.7之前我们会这样写:

SELECT `id` AS `id`,`name` FROM `game` GROUP BY `name`
  • 1

但是在5.7中,需要使用聚合函数:

SELECT ANY_VALUE(`id`) AS `id`,MAX(`id`),`name` FROM `game` GROUP BY `name`
  • 1

三、sql_mode解析:

参考自:https://blog.youkuaiyun.com/Cooldiok/article/details/59131952

ONLY_FULL_GROUP_BY: 对于GROUP BY聚合操作,如果在SELECT中的列,没有在GROUP
BY中出现,那么这个SQL是不合法的,因为列不在GROUP BY从句中

NO_AUTO_VALUE_ON_ZERO: 该值影响自增长列的插入。默认设置下,插入0或NULL代表生成下一个自增长值。如果用户
希望插入的值为0,而该列又是自增长的,那么这个选项就有用了。

STRICT_TRANS_TABLES: 在该模式下,如果一个值不能插入到一个事务表中,则中断当前的操作,对非事务表不做限制
NO_ZERO_IN_DATE: 在严格模式下,不允许日期和月份为零

NO_ZERO_DATE: 设置该值,mysql数据库不允许插入零日期,插入零日期会抛出错误而不是警告。

ERROR_FOR_DIVISION_BY_ZERO: 在INSERT或UPDATE过程中,如果数据被零除,则产生错误而非警告。如
果未给出该模式,那么数据被零除时MySQL返回NULL

NO_AUTO_CREATE_USER: 禁止GRANT创建密码为空的用户

NO_ENGINE_SUBSTITUTION:
如果需要的存储引擎被禁用或未编译,那么抛出错误。不设置此值时,用默认的存储引擎替代,并抛出一个异常

PIPES_AS_CONCAT:
将”||”视为字符串的连接操作符而非或运算符,这和Oracle数据库是一样的,也和字符串的拼接函数Concat相类似

ANSI_QUOTES: 启用ANSI_QUOTES后,不能用双引号来引用字符串,因为它被解释为识别符

 https://blog.youkuaiyun.com/u014520745/article/details/76056170

### 如何使用 Navicat 修改 MySQLsql_mode 配置 在 Navicat 中修改 MySQL 的 `sql_mode` 设置,可以按照以下方法操作: #### 1. 查询当前的 SQL Mode 为了确认当前的 `sql_mode` 值,在 Navicat 中执行以下 SQL 语句来获取当前服务器的模式设置: ```sql SELECT @@GLOBAL.sql_mode; ``` 这将返回全局范围内的 `sql_mode` 设置[^2]。 如果需要查看当前会话中的 `sql_mode`,可运行以下命令: ```sql SELECT @@SESSION.sql_mode; ``` #### 2. 修改 Global Sql_Mode 要永久更改 MySQL 的 `sql_mode` 并使其在数据库重启后仍然有效,需编辑 MySQL 的配置文件(通常是 my.cnf 或 my.ini)。然而,也可以通过 Navicat 执行 SQL 来临时调整全局 `sql_mode`。例如,禁用 `ONLY_FULL_GROUP_BY` 模式,可以运行以下命令: ```sql SET GLOBAL sql_mode = 'STRICT_TRANS_TABLES,NO_ZERO_IN_DATE,NO_ZERO_DATE,ERROR_FOR_DIVISION_BY_ZERO,NO_ENGINE_SUBSTITUTION'; ``` 上述命令移除了默认启用的 `ONLY_FULL_GROUP_BY` 模式,并保留其他推荐的安全选项[^3]。 注意:此方式仅适用于具有管理员权限的用户,且不会持久化到下次服务启动。 #### 3. 修改 Session Sql_Mode 对于不需要影响整个系统的场景,可以在单一会话中动态修改 `sql_mode`。例如: ```sql SET SESSION sql_mode = 'STRICT_TRANS_TABLES,NO_ZERO_IN_DATE,NO_ZERO_DATE,ERROR_FOR_DIVISION_BY_ZERO,NO_ENGINE_SUBSTITUTION'; ``` 这种方式只会影响当前连接下的行为,关闭连接后即失效[^4]。 #### 4. 编辑 MySQL 配置文件实现持久化变更 若希望长期生效,则应手动编辑 MySQL 的配置文件 (my.cnf),并添加或更新 `[mysqld]` 下的 `sql_mode` 参数。完成后记得重启 MySQL 服务使改动生效[^1]。 --- ### 注意事项 - 如果直接通过 SET 命令修改 global sql_mode 后遇到权限不足的问题,请联系 DBA 授予适当权限或者由其协助完成。 - 调整 sql_mode 可能会对现有查询逻辑造成潜在风险,请务必测试新设定下应用的表现是否正常。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值