当前位置: 首页 > news >正文

详细分析Mysql中的SQL_MODE基本知识(附Demo讲解)

目录

  • 前言
  • 1. 基本知识
  • 2. Demo讲解
    • 2.1 ONLY_FULL_GROUP_BY
    • 2.2 STRICT_TRANS_TABLES
    • 2.3 NO_ZERO_IN_DATE
    • 2.4 NO_ENGINE_SUBSTITUTION
    • 2.5 ANSI_QUOTES

前言

了解Mysql内部的机制有助于辅助开发以及形成整体的架构思维

对于基本的命令行以及优化推荐阅读:

  1. 数据库中增删改常用语法语句(全)
  2. Mysql优化高级篇(全)

1. 基本知识

SQL_MODE 是 MySQL 中用于设置 SQL 语法和行为的系统变量

控制 MySQL 的 SQL 解析和执行的方式,使其与 SQL 标准或其他数据库系统的行为一致
通过设置 SQL_MODE,可以改变 MySQL 处理特定 SQL 操作的方式

  • MySQL 5.7
    默认 SQL_MODE 包括:ONLY_FULL_GROUP_BY, STRICT_TRANS_TABLES, NO_ZERO_IN_DATE, NO_ZERO_DATE, ERROR_FOR_DIVISION_BY_ZERO, NO_AUTO_CREATE_USER, and NO_ENGINE_SUBSTITUTION

  • MySQL 8.0
    默认 SQL_MODE 包括:ONLY_FULL_GROUP_BY, STRICT_TRANS_TABLES, NO_ZERO_IN_DATE, NO_ZERO_DATE, ERROR_FOR_DIVISION_BY_ZERO, NO_ENGINE_SUBSTITUTION

相比之下,MySQL 8.0 中没有太大变化,但 NO_AUTO_CREATE_USER 被移除了,因为 MySQL 8.0 取消了自动创建用户的特性

2. Demo讲解

常见的 SQL_MODE 设置和示例,以帮助理解其影响

2.1 ONLY_FULL_GROUP_BY

控制对于 GROUP BY 子句的处理方式
在默认情况下,MySQL 允许在 SELECT 查询中使用 GROUP BY 子句时,对于不在 GROUP BY 子句中的非聚合列进行隐式处理,可能导致意外的结果

ONLY_FULL_GROUP_BY 关闭的情况下,MySQL可能会随意选择一行来代表每个分组,而不是严格按照 SQL 标准进行操作,如下:

 this is incompatible with sql_mode=only_full_group_by

截图如下:

在这里插入图片描述

对此设置为ONLY_FULL_GROUP_BY 模式

-- 开启 ONLY_FULL_GROUP_BY 模式
SET sql_mode = 'ONLY_FULL_GROUP_BY';-- 查询每个学生的平均成绩
SELECT name, AVG(salary) AS avg_salary FROM employees GROUP BY name;

在这里插入图片描述

2.2 STRICT_TRANS_TABLES

STRICT_TRANS_TABLES 模式下,如果插入的数据有问题(如超出字段长度或类型不匹配),MySQL 会报错并回滚事务

-- 设置 SQL_MODE 为 STRICT_TRANS_TABLES
SET sql_mode = 'STRICT_TRANS_TABLES';-- 创建表
CREATE TABLE demo_strict (id INT,name VARCHAR(5)
);-- 尝试插入超长数据
INSERT INTO demo_strict VALUES (1, 'TooLongName');

执行结果下:

INSERT INTO demo_strict VALUES (1, 'TooLongName');
[Err] 1406 - Data too long for column 'name' at row 1

截图如下:

在这里插入图片描述

2.3 NO_ZERO_IN_DATE

-- 设置 SQL_MODE 为 NO_ZERO_IN_DATE, NO_ZERO_DATE
SET sql_mode = 'NO_ZERO_IN_DATE,NO_ZERO_DATE';-- 创建表
CREATE TABLE demo_date (id INT,date_field DATE
);-- 尝试插入无效日期
INSERT INTO demo_date VALUES (1, '2020-00-00');

但在执行的过程中是可以成功的,但查询的时候日期显示未0000-00-00
为了避免这种插入无效日期,应该更改为:

-- 确保启用严格模式和日期模式
SET sql_mode = 'STRICT_TRANS_TABLES,NO_ZERO_IN_DATE,NO_ZERO_DATE';-- 创建表
CREATE TABLE demo_date (id INT,date_field DATE
);-- 尝试插入无效日期
INSERT INTO demo_date VALUES (1, '2020-00-00');

最终结果如下:

[Err] 1292 - Incorrect date value: '2020-00-00' for column 'date_field' at row 1

截图如下:

在这里插入图片描述
确保输出包含 STRICT_TRANS_TABLES 和日期相关的模式:

STRICT_TRANS_TABLES,NO_ZERO_IN_DATE,NO_ZERO_DATE,...

2.4 NO_ENGINE_SUBSTITUTION

指定的存储引擎不可用,则会报错,而不是使用默认的存储引擎

-- 设置 SQL_MODE 为 NO_ENGINE_SUBSTITUTION
SET sql_mode = 'NO_ENGINE_SUBSTITUTION';-- 尝试创建不存在的存储引擎的表
CREATE TABLE demo_engine (id INT
) ENGINE=NON_EXISTENT_ENGINE;

报错结果如下:[Err] 1286 - Unknown storage engine 'NON_EXISTENT_ENGINE'

截图如下:

在这里插入图片描述

2.5 ANSI_QUOTES

双引号用于标识符,而不是字符串

-- 设置 SQL_MODE 为 ANSI_QUOTES
SET sql_mode = 'ANSI_QUOTES';-- 尝试使用双引号作为标识符
CREATE TABLE "demo_quotes" ("id" INT,"name" VARCHAR(50)
);-- 插入数据
INSERT INTO "demo_quotes" ("id", "name") VALUES (1, 'John Doe');

截图如下:

在这里插入图片描述

成功插入

http://www.lryc.cn/news/364668.html

相关文章:

  • vue3+uniapp
  • 组织病理学结合人工智能之后,如何实际应用于临床?|顶刊精析·24-06-06
  • VCAST创建单元测试工程
  • 数据结构之归并排序算法【图文详解】
  • 设计模式基础
  • Glide支持通过url加载本地图标
  • 网络安全形势与WAF技术分享
  • 【实战JVM】-实战篇-06-GC调优
  • 深入解析智慧互联网医院系统源码:医院小程序开发的架构到实现
  • 获取 Bean 对象更加简单的方式
  • ChatGPT基本原理
  • 几种更新 npm 项目依赖的实用方法
  • Python爬虫之简单学习BeautifulSoup库,学习获取的对象常用方法,实战豆瓣Top250
  • SAP-BASIS15-查看系统状态
  • 前端怎么debugger排查线上问题
  • LabVIEW源程序安全性保护综合方案
  • JS包装类:循环中为什么建议用变量存储str.length进行循环判断?
  • Android Audio实战——音量默认值修改(一)
  • 解决uni-app progress控件不显示问题
  • 使用C++版本的opencv dnn 部署onnx模型
  • python中实现队列功能
  • 自然资源-关于城镇开发边界局部优化的政策思路梳理
  • ElementUI的Table组件在无数据情况下让“暂无数据”文本居中显示
  • SAP-BASIS14-安装语言包
  • ant design的upload组件踩坑记录
  • Python私教张大鹏 Vue3整合AntDesignVue之按钮组件
  • 【小海实习日记】PHP安装
  • C++ Primer Chapter 4 Expressions
  • [leetcode hot 150]第一百三十七题,只出现一次的数字Ⅱ
  • wpf工程中加入Hardcodet.NotifyIcon.Wpf生成托盘