2023-05-10 08:00:02

禁用ONLY_FUL_GROUP_BY

我意外启用了ONLY_FUL_GROUP_BY模式,如下所示:

SET sql_mode = 'ONLY_FULL_GROUP_BY';

如何禁用它?


当前回答

在MySQL 5.7和Ubuntu 16.04上,编辑文件MySQL.cnf。

$ sudo nano /etc/mysql/conf.d/mysql.cnf

包括如下所示的sql_mode并保存文件。

[mysql]
sql_mode=IGNORE_SPACE,NO_ZERO_IN_DATE,NO_ZERO_DATE,ERROR_FOR_DIVISION_BY_ZERO,NO_AUTO_CREATE_USER,NO_ENGINE_SUBSTITUTION

注意,在我的例子中,我删除了模式STRICT_TRANS_TABLES和ONLY_FUL_GROUP_BY。

这样做,它将永久保存模式配置。如果您只是通过MySQL更新@@sql_mode,则会有所不同,因为它会在机器/服务重新启动时重置。

之后,要使修改后的配置生效,请重新启动mysql服务:

$ sudo service mysql restart

尝试访问mysql:

$ mysql -u user_name -p

如果您能够登录并访问MySQL控制台,那就好了。太好了!

但是,如果像我一样,您会遇到错误“未知变量sql_mode”,这表明sql_mode是mysqld的一个选项,那么您必须返回,再次编辑文件mysql.cnf并将[mysql]更改为[mysqld]。重新启动MySQL服务并进行最后一次测试,尝试登录MySQL控制台。给你!

其他回答

解决方案1:从mysql控制台中删除ONLY_FUL_GROUP_BY

mysql > SET GLOBAL sql_mode=(SELECT REPLACE(@@sql_mode,'ONLY_FULL_GROUP_BY',''));

你可以在这里阅读更多

解决方案2:从phpmyadmin中删除ONLY_FUL_GROUP_BY

打开phpmyadmin并选择localhost单击菜单变量并向下滚动以进入sql模式单击编辑按钮更改值并删除ONLY_FUL_GROUP_BY并单击保存。

    mysql> set global sql_mode='STRICT_TRANS_TABLES,NO_ZERO_IN_DATE,NO_ZERO_DATE,ERROR_FOR_DIVISION_BY_ZERO,NO_AUTO_CREATE_USER,NO_ENGINE_SUBSTITUTION';
    mysql> set session sql_mode='STRICT_TRANS_TABLES,NO_ZERO_IN_DATE,NO_ZERO_DATE,ERROR_FOR_DIVISION_BY_ZERO,NO_AUTO_CREATE_USER,NO_ENGINE_SUBSTITUTION';
    mysql> exit;

向sql_mode添加或删除模式

MySQL 5.7.9或更高版本

要在sql_mode中添加或删除模式,可以使用list_add和list_drop函数。

要从当前SESSION.sql_mode中删除模式,可以使用以下方法之一:

SET SESSION sql_mode = sys.list_drop(@@SESSION.sql_mode, 'ONLY_FULL_GROUP_BY');
SET sql_mode = sys.list_drop(@@sql_mode, 'ONLY_FULL_GROUP_BY');
SET @@sql_mode = sys.list_drop(@@sql_mode, 'ONLY_FULL_GROUP_BY');

从GLOBAL.sql_mode中删除当前运行时操作持续的模式,直到服务重新启动。

SET GLOBAL sql_mode = sys.list_drop(@@GLOBAL.sql_mode, 'ONLY_FULL_GROUP_BY');

MySQL 5.7.8或更高版本

由于sql_mode值是一个CSV模式字符串,因此您需要确保该字符串不包含残留逗号,这可以通过使用TRIM(BOTH','FROM…)来实现。

要从sql_mode变量中删除模式,您需要使用REPLACE()和TRIM()来确保删除任何残留的逗号。

SET SESSION sql_mode = TRIM(BOTH ',' FROM REPLACE(@@SESSION.sql_mode, 'ONLY_FULL_GROUP_BY', ''));
SET GLOBAL sql_mode = TRIM(BOTH ',' FROM REPLACE(@@GLOBAL.sql_mode, 'ONLY_FULL_GROUP_BY', ''));

要将模式添加到sql_mode变量,您需要使用CONCAT_WS(',',…)来确保在当前模式后面附加逗号,并使用TRIM()来确保删除任何剩余的逗号。

SET SESSION sql_mode = TRIM(BOTH ',' FROM CONCAT_WS(',', 'ONLY_FULL_GROUP_BY', @@SESSION.sql_mode));
SET GLOBAL sql_mode = TRIM(BOTH ',' FROM CONCAT_WS(',', 'ONLY_FULL_GROUP_BY', @@GLOBAL.sql_mode));

注意:在建立新连接之前,更改GLOBAL变量不会传播到SESSION变量。GLOBAL变量将持续存在,直到正在运行的服务重新启动。SESSION变量将持续用于当前连接,直到连接关闭并建立新连接。


还原为GLOBAL.sql_mode

由于SET sql_mode='ONLY_FUL_GROUP_BY';在没有GLOBAL修饰符的情况下执行时,更改仅影响当前SESSION状态值,该值也属于@@sql_mode。要删除它并恢复到服务器重启时的全局默认值,您需要使用@@global.sql_mode中的值。[sic]

当前SESSION值仅对当前连接有效。重新连接到服务器将将值还原回GLOBAL价值

要将当前会话状态值还原为当前全局值,可以使用以下方法之一:

SET SESSION sql_mode = @@GLOBAL.sql_mode;
SET @@sql_mode = @@GLOBAL.sql_mode;
SET sql_mode = @@GLOBAL.sql_mode;

将SESSION.sql_mode值更改为ONLY_FUL_GROUP_BY

SET sql_mode = 'ONLY_FULL_GROUP_BY';
SELECT @@sql_mode, @@GLOBAL.sql_mode;
+--------------------+----------------------------------------------+
| @@sql_mode         | @@GLOBAL.sql_mode                            |
+--------------------+----------------------------------------------+
| ONLY_FULL_GROUP_BY | NO_AUTO_VALUE_ON_ZERO,NO_ENGINE_SUBSTITUTION |
+--------------------+----------------------------------------------+

将SESSION.sql_mode值还原为GLOBAL.sql_mde值

SET sql_mode = @@GLOBAL.sql_mode;
SELECT @@sql_mode, @@GLOBAL.sql_mode;
+----------------------------------------------+----------------------------------------------+
| @@sql_mode                                   | @@GLOBAL.sql_mode                            |
+----------------------------------------------+----------------------------------------------+
| NO_AUTO_VALUE_ON_ZERO,NO_ENGINE_SUBSTITUTION | NO_AUTO_VALUE_ON_ZERO,NO_ENGINE_SUBSTITUTION |
+----------------------------------------------+----------------------------------------------+

使用选项文件重新启动服务器持久sql_mode

设置SQL模式在服务器启动时,在命令上使用--sqlmode=“modes”选项行或sql mode=“modes”(在选项文件中,如my.cnf(Unix操作系统)或my.ini(Windows)。[原文]

请查看您的MySQL版本,以确定支持的模式和默认模式。

MySQL>=5.7.5,<=5.7.6默认值

[mysqld]
    
sql-mode="ONLY_FULL_GROUP_BY,STRICT_TRANS_TABLES,NO_ENGINE_SUBSTITUTION"

有关详细信息,请参阅选项文件语法。

在选项文件中指定选项的语法类似于命令行语法。但是,在选项文件中,可以省略前导选项名称中的两个破折号,每个选项只能指定一个选项线例如,命令行上的--quick和--host=localhost应在选项文件。要在中指定格式为--loose-opt_name的选项一个选项文件,将其写为loose-opt_name。

该值可选地可以用单引号括起来,或者双引号,如果值包含#注释字符。

默认sql_mode值

由于删除了每个版本的MySQL文档值,我将它们添加到这里供您参考。

MySQL 8.0[sic]MySQL 5.7[sic]MySQL 5.6[sic]

MySQL>=8.0.11 8.0.5-8.0.10已跳过

ONLY_FULL_GROUP_BY STRICT_TRANS_TABLES NO_ZERO_IN_DATE NO_ZERO_DATE ERROR_FOR_DIVISION_BY_ZERO NO_ENGINE_SUBSTITUTION

MySQL>=5.7.8,<=8.0.4

ONLY_FULL_GROUP_BY STRICT_TRANS_TABLES NO_ZERO_IN_DATE NO_ZERO_DATE ERROR_FOR_DIVISION_BY_ZERO NO_AUTO_CREATE_USER NO_ENGINE_SUBSTITUTION

MySQL 5.7.7

ONLY_FULL_GROUP_BY STRICT_TRANS_TABLES NO_AUTO_CREATE_USER NO_ENGINE_SUBSTITUTION

MySQL>=5.7.5,<=5.7.6

ONLY_FULL_GROUP_BY STRICT_TRANS_TABLES NO_ENGINE_SUBSTITUTION

MySQL>=5.6.6,<=5.7.4

NO_ENGINE_SUBSTITUTION

MySQL<=5.6.5

''

最好是在新项目上使用ONLY_FUL_GROUP_BY,同时在现有站点和数据库上保持向后兼容。为此,我选择在类中加载mysqli和PDO时修改SQL模式。

对于mysqli:

mysqli_query($this->link,"SET SESSION sql_mode=(SELECT REPLACE(@@sql_mode,'ONLY_FULL_GROUP_BY',''))");

对于PDO类:

PDO::MYSQL_ATTR_INIT_COMMAND => "SET SESSION sql_mode=(SELECT REPLACE(@@sql_mode,'ONLY_FULL_GROUP_BY',''))"

PDO初始化示例:

try {
    $dns     = 'mysql:host=localhost;dbname=' . $prod_db;
    $user    = _DBUSER_;
    $pass    = _DBPASS_;
    $options = array(
        PDO::MYSQL_ATTR_INIT_COMMAND => "SET NAMES utf8",
        PDO::MYSQL_ATTR_INIT_COMMAND => "SET SESSION sql_mode=(SELECT REPLACE(@@sql_mode,'ONLY_FULL_GROUP_BY',''))"
    );

    $db = new PDO($dns, $user, $pass, $options);
} 

catch(Exception $e) {
    mail("contact@xxxxxx.xx","Database Error ",$dns.",".$user);
    echo "Unable ot connect to mySQL : ", $e->getMessage();
    die();
} 

因此,在同一服务器上创建的新类将使用新的数据库标准。

使用MySQL 5.7.28版检查SELECT@@sql_mode;并使用更新

SET @@sql_mode = 'STRICT_TRANS_TABLES,NO_ZERO_IN_DATE,NO_ZERO_DATE,ERROR_FOR_DIVISION_BY_ZERO,NO_AUTO_CREATE_USER,NO_ENGINE_SUBSTITUTION'