【MySQL】sql_mode引起的一个问题和总结

【背景】

  之前项目中,项目组计划将现场的MySQL5.5升级到5.7,以提升主从同步性能、使用半同步复制,以及解决一些现场问题等。安排测试组进行验证,测试同事反馈实验室环境中发现有入库失败,我查看了error_log日志,发现有不少如下报错。

[Err] 1364 - Field `xx_field` doesn't have a default value

 

【排查与分析】

  业务版本前后都是一样的,好端端的mysql怎么突然就部分表写入失败呢?根据上面的日志很快猜到是 sql_mode 问题: NOT NULL 列没有默认值但代码里也没给值,在非严格模式下,int列默认为0,string列默认为''了,所以不成问题;但在严格模式下,是直接返回失败的。

       那么看看吧,果然如此。

mysql> show variables like "sql_mode";
+---------------+--------------------------------------------+
| Variable_name | Value                                      |
+---------------+--------------------------------------------+
| sql_mode      | STRICT_TRANS_TABLES,NO_ENGINE_SUBSTITUTION |
+---------------+--------------------------------------------+

  但测试同事反馈并没有更改过该参数。查阅资料,发现:MySQL5.6.6 以后版本默认就是NO_ENGINE_SUBSTITUTION,STRICT_TRANS_TABLES,5.5默认为 '' 。

  

【解决】  

  严格模式是合理的,更能保证系统的健壮性,所以解决方案并不是set global variables去除STRICT_TRANS_TABLES,而是业务侧修改完善,加上默认值。

 

【总结】

  凡事讲究举一反三,PDCA。

  • 本次出现问题的STRICT_TRANS_TABLES模式。 

  严格模式,进行数据的严格校验,错误数据不能插入,报error错误。 

  单独指 INSERTUPDATE出现少值或无效值该如何处理: 

  1. '' 传给int,严格模式下非法,若启用非严格模式则变成0,产生一个warning
  2. Out Of Range,变成插入最大边界值
  3. 非null字段没有默认值,插入记录时该字段缺失时报错。A value is missing when a new row to be inserted does not contain a value for a non-NULL column that has no explicit DEFAULT clause in its definition

 

  • sql_mode取值很多,官方有打包组合,ANSI、TRADITIONAL,当然,每项也可以单独设置。注:大小写不敏感,都可以。

1. sql_mode='ANSI'

ANSI模式:宽松模式,对插入数据进行校验,如果不符合定义类型或长度,对数据类型调整或截断保存,报warning警告。

更改语法和行为,使其更符合标准SQL,相当于REAL_AS_FLOAT, PIPES_AS_CONCAT, ANSI_QUOTES, IGNORE_SPACE。

2.sql_mode='TRADITIONAL'

TRADITIONAL 模式:严格模式,当向mysql数据库插入数据时,进行数据的严格校验,保证错误数据不能插入,报error错误。用于事物时,会进行事物的回滚。

更像传统SQL数据库系统,该模式的简单描述是当在列中插入不正确的值时“给出错误而不是警告”。

相当于 STRICT_TRANS_TABLES, STRICT_ALL_TABLES, NO_ZERO_IN_DATE, NO_ZERO_DATE, ERROR_FOR_DIVISION_BY_ZERO, NO_AUTO_CREATE_USER, NO_ENGINE_SUBSTITUTION。 

  • 其他常用

set session sql_mode = 'no_auto_create_user';

MySQL5.7及之前版本可设置,MySQL8之后为默认设置,先create user才能grant。 

 

set session sql_mode = 'no_zero_in_date';

set session sql_mode = 'no_zero_date';

no_zero_date认为日期 '0000-00-00' 非法,与是否设置后面的严格模式有关。

  1. 如果设置了严格模式,则 NO_ZERO_DATE 自然满足。但如果是 INSERT IGNORE 或 UPDATE IGNORE,'0000-00-00'依然允许且只显示warning
  2. 如果在非严格模式下,设置了NO_ZERO_DATE,效果与上面一样,'0000-00-00'允许但显示warning;如果没有设置NO_ZERO_DATE,no warning,当做完全合法的值。
  3. NO_ZERO_IN_DATE情况与上面类似,不同的是控制日期和天,是否可为 0 ,即 2010-01-00 是否合法。

 

set session sql_mode = 'no_engine_substitution';

使用 ALTER TABLE或CREATE TABLE 指定 ENGINE 时, 需要的存储引擎被禁用或未编译,该如何处理。

启用NO_ENGINE_SUBSTITUTION时,此时直接抛出错误;

不设置此值时,CREATE用默认的存储引擎替代,ATLER不进行更改,并抛出一个 warning。

转载于:https://www.cnblogs.com/wwcom123/p/10715285.html

<think>好的,我需要解决用户在Docker中运行的MySQL遇到的sql_mode包含only_full_group_by导致的问题。首先,我得回忆一下MySQLsql_mode的作用,特别是only_full_group_by模式的影响。这个模式要求GROUP BY子句必须包含所有非聚合列,否则会报错。用户可能是在执行某些查询时遇到了错误,需要修改或禁用这个配置。 接下来,用户使用的是Docker环境,所以需要找到如何在Docker容器中修改MySQL配置的方法。通常,MySQL的配置文件是my.cnf,位于/etc/mysql目录下。在Docker中,可以通过挂载自定义的配置文件到容器内的对应位置来覆盖默认配置。 然后,考虑如何具体修改sql_mode。用户可能想完全移除only_full_group_by,或者调整其他模式。需要给出修改my.cnf的示例,比如在[mysqld]部分设置sql_mode为其他值,比如STRICT_TRANS_TABLES,NO_ENGINE_SUBSTITUTION。同时,要提醒用户修改后重启MySQL服务或者整个容器才能使配置生效。 另外,用户可能不想修改配置文件,而是临时在运行时修改sql_mode。这时候可以通过进入容器执行SQL命令来动态调整,但要注意这种修改在容器重启后会失效。需要提供相应的命令示例,比如SET GLOBAL sql_mode = '...';,并说明这种方法的临时性。 还要考虑用户可能使用的Docker运行命令是否正确挂载了配置文件,比如使用-v参数将宿主机的配置文件目录挂载到容器的/etc/mysql/conf.d。需要检查用户是否已经正确设置了挂载点,或者是否需要创建自定义的配置文件。 另外,可能需要验证配置是否生效,比如进入容器查看sql_mode的当前值,或者执行查询测试是否还会报错。提供相应的检查步骤,比如执行SELECT @@GLOBAL.sql_mode;命令。 最后,需要确保引用的参考资料相关,虽然用户提供的引用似乎MySQL或Docker关系不大,但可能涉及配置文件的处理方式,可以适当关联。例如,引用中提到配置用户组提供程序,可能与配置文件修改有关联,但需要谨慎引用,避免误导用户。 总结步骤应该是:修改配置文件、挂载到容器、重启服务、验证配置。同时提供临时方法持久化方法的区别,确保用户理解不同方案的适用场景。</think>### Docker中MySQL禁用only_full_group_by配置解决方案 在Docker容器中运行的MySQL如果启用了$sql_mode=ONLY_FULL_GROUP_BY$,会导致包含非聚合列的GROUP BY查询报错。以下是两种主要解决方法: #### 方法一:修改配置文件(推荐) 1. **创建自定义配置文件** 新建文件$mysql-custom.cnf$,内容为: ```ini [mysqld] sql_mode=STRICT_TRANS_TABLES,NO_ZERO_IN_DATE,NO_ZERO_DATE,ERROR_FOR_DIVISION_BY_ZERO,NO_ENGINE_SUBSTITUTION ``` 2. **启动容器时挂载配置文件** ```bash docker run -d \ --name mysql8 \ -v /path/to/mysql-custom.cnf:/etc/mysql/conf.d/mysql-custom.cnf \ -e MYSQL_ROOT_PASSWORD=your_password \ mysql:8.0 ``` 3. **验证配置** ```bash docker exec -it mysql8 mysql -uroot -p SELECT @@GLOBAL.sql_mode; ``` #### 方法二:运行时动态修改(临时) ```sql -- 进入MySQL命令行 SET GLOBAL sql_mode=(SELECT REPLACE(@@sql_mode,'ONLY_FULL_GROUP_BY','')); ``` > 注意:该方法在容器重启后失效,需结合方法一实现持久化[^1] #### 配置原理说明 MySQL的$sql_mode$参数控制SQL语法校验严格程度,$ONLY_FULL_GROUP_BY$是SQL标准中GROUP BY子句的严格模式。通过修改配置文件可以永久改变服务端配置,而动态修改只影响当前会话[^2]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值