MySQL(高级特性篇) 01 章——Linux下MySQL的安装与使用
一、安装前说明
(1)Linux系统及工具的准备
- 安装并启动好两台虚拟机:CentOS 7
- 掌握克隆虚拟机的操作:
- MAC地址(点击生成,确保两台虚拟机的MAC地址不一样)
- 主机名(如果是只读文件,可以先通过root用户运行:即在终端输入su root,再对文件修改)重启以后,可以看到主机名已经改变
vim /etc/hostname
- ip地址
vim /etc/sysconfig/network-scripts/ifcfg-ens33
- UUID
- 安装有XShell和Xftp等访问CentOS系统的工具
- CentOS6和CentOS7在MySQL的使用中的区别:
- 防火墙:6是iptables,7是firewalld
- 启动服务的命令:6是service,7是systemctl
(2)查看是否安装过MySQL
- 如果使用rpm安装,检查一下RPM PACKAGE:
rpm -qa | grep -i mysql # -i 忽略大小写
- 检查mysql service:
systemctl status mysqld.service
- 如果存在mysql-libs的旧版本包,显示如下:
- 如果不存在mysql-libs的旧版本包,显示如下:
(3)MySQL的卸载
- 关闭mysql服务:
systemctl stop mysqld.service
- 查看当前mysql安装状况:
rpm -qa | grep -i mysql # 或 yum list installed | grep mysql
- 卸载上述命令查询出的已安装程序(务必卸载干净,反复执行rpm -qa | grep -i mysql确认是否有卸载残留):
yum remove mysql-xxx mysql-xxx mysql-xxx mysqk-xxxx
- 删除mysql相关文件:
- 查找相关文件:
find / -name mysql
- 删除上述命令查找出的相关文件:
rm -rf xxx
- 查找相关文件:
- 删除my.cnf:
rm -rf /etc/my.cnf
二、MySQL的Linux版安装
(1)MySQL的4大版本
- MySQL Community Server 社区版本:开源免费,自由下载,但不提供官方技术支持,适用于大多数普通用户
- MySQL Enterprise Edition 企业版本:需付费,不能在线下载,可以试用30天。提供了更多的功能和更完备的技术支持,更适合于对数据库的功能和可靠性要求较高的企业客户
- MySQL Cluster 集群版:开源免费,用于架设集群服务器,可以将几个MySQL Server封装成一个Server。需要在社区版或企业版的基础上使用
- MySQL Cluster CGE 高级集群版:需付费
- 截至目前,官方最新版本为8.0.27
- 此外,官方还提供了MySQL WorkBench,这是一款专为MySQL设计的ER/数据库建模工具。MySQL WorkBench又分为两个版本,分别是社区版和商用版
(2)下载MySQL指定版本
- Linux系统下安装软件用的第一种方式:使用rpm命令安装扩展名为“.rpm”的软件包
- Linux系统下安装软件用的第二种方式:yum命令,需联网,从互联网获取的yum源,直接使用yum命令安装
- Linux系统下安装软件用的第三种方式:针对tar.gz这样的压缩格式,要用tar命令来解压;如果是其它压缩格式,就使用其它命令
- Linux系统下安装MySQL,官方给出多种安装方式
安装方式 特点 rpm 安装简单,灵活性差,无法灵活选择版本、升级 rpm repository 安装包极小,版本安装简单灵活,升级方便,需要联网安装 通用二进制包 安装比较复杂,灵活性高,平台通用性好 源码包 安装最复杂,时间长,参数设置灵活,性能好 - 下载Linux下的MySQL(将tar文件解压):MySQL :: Download MySQL Community Server (Archived Versions)
- 抽取出来的安装包(8.0版本)
- mysql-community-client-8.0.25-1.el7.x86_64.rpm
- mysql-community-client-plugins-8.0.25-1.el7.x86_64.rpm
- mysql-community-common-8.0.25-1.el7.x86_64.rpm
- mysql-community-libs-8.0.25-1.el7.x86_64.rpm
- mysql-community-server-8.0.25-1.el7.x86_64.rpm
- 从解压后的文件夹里抽取出要用的zip
- 抽取出来的安装包(5.7版本)
- mysql-community-client-5.7.28-1.el7.x86_64.rpm
- mysql-community-common-5.7.28-1.el7.x86_64.rpm
- mysql-community-libs-5.7.28-1.el7.x86_64.rpm
- mysql-community-server-5.7.28-1.el7.x86_64.rpm
- 从解压后的文件夹里抽取出要用的zip
- 将抽取出来的安装包用Xftp上传到对应CentOS的/opt文件夹中
(3)CentOS7下检查MySQL依赖
2.3.1检查/tmp临时目录权限(必不可少)
- 由于MySQL安装过程中,会通过MySQL用户在/tmp目录下新建tmp_db文件,所以请给/tmp较大的权限。执行:
chmod -R 777 /tmp
2.3.2安装前,检查依赖
- 命令:
rpm -qa|grep libaio
- 如果存在libaio包如下:
- 命令:
rpm -qa|grep net-tools
- 如果存在net-tools包如下:
- 只要正常安装,这两个包都是有的
(4)CentOS7下MySQL安装过程
- 将安装程序拷贝到/opt目录下后,我们在MySQL的安装文件目录下执行(必须按照顺序执行):
- 代码:
rpm -ivh mysql-community-common-8.0.25-1.el7.x86_64.rpm rpm -ivh mysql-community-client-plugins-8.0.25-1.el7.x86_64.rpm rpm -ivh mysql-community-libs-8.0.25-1.el7.x86_64.rpm rpm -ivh mysql-community-client-8.0.25-1.el7.x86_64.rpm rpm -ivh mysql-community-server-8.0.25-1.el7.x86_64.rpm
- 注意: 如在检查工作时,没有检查mysql依赖环境在安装mysql-community-server会报错
- rpm是Redhat Package Manage缩写,通过RPM的管理,用户可以把源代码包装成以rpm为扩展名的 文件形式,易于安装
- -i:--install 安装软件包
- -v:--verbose 提供更多的详细信息输出
- -h:--hash 软件包安装的时候列出哈希标记 (和 -v 一起使用效果更好),展示进度条
- 代码:
- 如果出现该错误,解决方式是:
yum remove mysql-libs
- 查看MySQL版本:
mysql --version #或 mysqladmin --version
- 执行如下命令,查看是否安装成功。需要增加 -i 不用去区分大小写,否则搜索不到:
rpm -qa|grep -i mysql
- 为了保证数据库目录与文件的所有者为mysql登录用户,如果你是以 root 身份运行mysql 服务,需要执行下面的命令初始化
mysqld --initialize --user=mysql
- 说明:--initialize 选项默认以“安全”模式来初始化,则会为 root 用户生成一个密码并将该密码标记为过期 ,登录后你需要设置一个新的密码。生成的临时密码会往日志中记录一份
- 查看密码(可以看到密码是C1gG_YEIOdwb):
cat /var/log/mysqld.log
- 启动MySQL,查看状态:
#加不加.service后缀都可以 启动:systemctl start mysqld.service 关闭:systemctl stop mysqld.service 重启:systemctl restart mysqld.service 查看状态:systemctl status mysqld.service
- 查看进程:
ps -ef | grep -i mysql
- 查看MySQL服务是否自启动(默认是enabled):
systemctl list-unit-files|grep mysqld.service
- 如不是enabled可以运行如下命令设置自启动
systemctl enable mysqld.service
- 如果希望不进行自启动,运行如下命令设置
systemctl disable mysqld.service
三、MySQL登录
(1)首次登录
- 首次登录,密码是随机分配的,看上面安装过程
(2)修改密码
- 在做具体操作之前,必须先修改密码
- 修改密码:
ALTER USER 'root'@'localhost' IDENTIFIED BY 'new_password';
(3)设置远程登录
3.3.1当前问题
- 在用SQLyog或Navicat中配置远程连接Mysql数据库时遇到如下报错信息,这是由于Mysql配置了不支持远程连接引起的
3.3.2确认网络
- 在远程机器上使用ping ip地址,保证网络畅通
- 在远程机器上使用telnet命令,保证端口号开放访问(给Telnet客户端打勾),会发现虚拟机的3306端口是关闭的
3.3.3关闭防火墙或开放端口
- 方式一:关闭防火墙(最好把Windows的也关闭)
#CentOS6 service iptables stop #CentOS7 systemctl start firewalld.service systemctl status firewalld.service systemctl stop firewalld.service #设置开机启用防火墙 systemctl enable firewalld.service #设置开机禁用防火墙 systemctl disable firewalld.service
- 方式二:开放端口
#查看开放的端口号 firewall-cmd --list-all #设置开放的端口号 firewall-cmd --add-service=http --permanent firewall-cmd --add-port=3306/tcp --permanent #重启防火墙 firewall-cmd --reload
3.3.4Linux下修改配置
- 在Linux系统MySQL下测试(可以看到root用户的当前主机配置信息为localhost):
use mysql; select Host,User from user;
- 修改Host为通配符:Host列指定了允许用户登录所使用的IP,比如user=root Host=192.168.1.1。这里的意思就是说root用户只能通过192.168.1.1的客户端去访问。 user=root Host=localhost,表示只能通过本机客户端去访问。而 % 是个 通配符 ,如果Host=192.168.1.%,那么就表示只要是IP地址前缀为“192.168.1.”的客户端都可以连接。如果 Host=% ,表示所有IP都有连接权限
- 注意:在生产环境下不能为了省事将host设置为%,这样做会存在安全问题,具体的设置可以根据生产 环境的IP进行设置
- 代码:
update user set host = '%' where user ='root';
- Host设置了“%”后便可以允许远程访问。Host修改完成后记得执行flush privileges使配置立即生效
flush privileges;
3.3.5测试
- 如果是 MySQL8.0版本,连接时还会出现错误码为2058的问题
- 解决方法是:Linux下 mysql -u root -p 登录你的mysql 数据库,然后执行这条SQL:
ALTER USER 'root'@'%' IDENTIFIED WITH mysql_native_password BY 'abc123';
- 如图可见,连接成功
四、MySQL8的密码强度评估(了解)
(1)MySQL不同版本设置密码(可能出现)
- MySQL5.7中:
- MySQL8.0中(可能出现):
mysql> alter user 'root' identified by 'abcd1234'; # HelloWorld_123 ERROR 1819 (HY000): Your password does not satisfy the current policy requirements
(2)MySQL8之前的安全策略
- 在MySQL 8.0之前,MySQL使用的是validate_password插件检测、验证账号密码强度,保障账号的安全性
- 安装/启用插件的方式一(不推荐):在参数文件my.cnf中添加参数(别忘了重启服务)
[mysqld] plugin-load-add=validate_password.so \#ON/OFF/FORCE/FORCE_PLUS_PERMANENT: 是否使用该插件(及强制/永久强制使用) validate-password=FORCE_PLUS_PERMANENT
- 安装/启用插件的方式二:运行时命令安装
INSTALL PLUGIN validate_password SONAME 'validate_password.so';
- 方式二也会注册到元数据,也就是mysql.plugin表中,所以不用担心MySQL重启后插件会失效
(3)MySQL8的安全策略
- MySQL 8.0,引入了服务器组件(Components)这个特性,validate_password插件已用服务器组件重新实 现。8.0.25版本的数据库中,默认自动安装validate_password组件
- 未安装插件前,执行如下两个指令:
- 安装插件后,执行如下两个指令:
- 关于 validate_password 组件对应的系统变量说明:
选项 默认值 参数描述 validate_password_check_user_name ON 设置为ON的时候表示能将密码设置成当前 用户名 validate_password_dictionary_file 用于检查密码的字典文件的路径名,默认为 空 validate_password_length 8 密码的最小长度,也就是说密码长度必须大 于或等于8 validate_password_mixed_case_count 1 如果密码策略是中等或更强的, validate_password要求密码具有的小写和大 写字符的最小数量。对于给定的这个值密码 必须有那么多小写字符和那么多大写字符 validate_password_number_count 1 密码必须包含的数字个数 validate_password_policy MEDIUM 密码强度检验等级,可以使用数值0、1、2 或相应的符号值LOW、MEDIUM、STRONG来 指定。 0/LOW :只检查长度。 1/MEDIUM :检查长度、数字、大小写、特殊字符。 2/STRONG :检查长度、数字、大 小写、特殊字符、字典文件 validate_password_special_char_count 1 密码必须包含的特殊字符个数 - 修改安全策略:
SET GLOBAL validate_password_policy=LOW; SET GLOBAL validate_password_policy=MEDIUM; SET GLOBAL validate_password_policy=STRONG; SET GLOBAL validate_password_policy=0; # For LOW SET GLOBAL validate_password_policy=1; # For MEDIUM SET GLOBAL validate_password_policy=2; # For HIGH #注意,如果是插件的话,SQL为set global validate_password_policy=LOW #此外,还可以修改密码中字符的长度 set global validate_password_length=1;
- 密码强度测试:如果你创建密码是遇到“Your password does not satisfy the current policy requirements”,可以通过函数组 件去检测密码是否满足条件: 0-100。当评估在100时就是说明使用上了最基本的规则:大写+小写+特殊字符+数字组成的8位以上密码
(4)卸载插件、组件(了解)
- 卸载插件
mysql> UNINSTALL PLUGIN validate_password; Query OK, 0 rows affected, 1 warning (0.01 sec)
- 卸载组件
mysql> UNINSTALL COMPONENT 'file://component_validate_password'; Query OK, 0 rows affected (0.02 sec)
五、字符集的设置
(1)修改MySQL5.7字符集
5.1.1修改步骤
- 在MySQL 8.0版本之前,默认字符集为 latin1 ,utf8字符集指向的是 utf8mb3 。网站开发人员在数据库 设计的时候往往会将编码修改为utf8字符集。如果遗忘修改默认的编码,就会出现乱码的问题。从MySQL 8.0开始,数据库的默认编码将改为 utf8mb4 ,从而避免上述乱码的问题
- 操作一:查看默认使用的字符集
- 代码:
show variables like 'character%'; # 或者 show variables like '%char%';
- 在MySQL8.0中执行:
- 在MySQL5.7中执行:
- 代码:
- 操作二:修改字符集(character_set_server=utf8)
vim /etc/my.cnf
- 重新启动MySQL服务(但是原库、原表的设定不会发生变化,参数修改只对新建的数据库生效)
5.1.2已有库&表字符集的变更
- MySQL5.7版本中,以前创建的库,创建的表字符集还是latin1
- 修改已创建数据库的字符集
alter database dbtest1 character set 'utf8';
- 修改已创建数据表的字符集
alter table t_emp convert to character set 'utf8';
- 注意:但是原有的数据如果是用非'utf8'编码的话,数据本身编码不会发生改变。已有数据需要导 出或删除,然后重新插入
(2)各级别的字符集
- MySQL有4个级别的字符集和比较规则,分别是:
- 服务器级别
- 数据库级别
- 表级别
- 列级别
- 执行如下SQL语句:
- 在上图中:
- character_set_server:服务器级别的字符集
- character_set_database:当前数据库的字符集
- character_set_client:服务器解码请求时使用的字符集
- character_set_connection:服务器处理请求时会把请求字符串从character_set_client转为 character_set_connection
- character_set_results:服务器向客户端返回数据时使用的字符集
5.2.1服务器级别
- character_set_server :服务器级别的字符集
- 我们可以在配置文件中这样写:
[server] character_set_server=gbk # 默认字符集 collation_server=gbk_chinese_ci #对应的默认的比较规则
- 当服务器启动的时候读取这个配置文件后这两个系统变量的值便修改了
5.2.2数据库级别
- character_set_database :当前数据库的字符集
- 我们在创建和修改数据库的时候可以指定该数据库的字符集和比较规则,具体语法如下:
CREATE DATABASE 数据库名 [[DEFAULT] CHARACTER SET 字符集名称] [[DEFAULT] COLLATE 比较规则名称]; ALTER DATABASE 数据库名 [[DEFAULT] CHARACTER SET 字符集名称] [[DEFAULT] COLLATE 比较规则名称];
5.2.3表级别
- 我们也可以在创建和修改表的时候指定表的字符集和比较规则,语法如下:
CREATE TABLE 表名 (列的信息) [[DEFAULT] CHARACTER SET 字符集名称] [COLLATE 比较规则名称]] ALTER TABLE 表名 [[DEFAULT] CHARACTER SET 字符集名称] [COLLATE 比较规则名称]
- 如果创建和修改表的语句中没有指明字符集和比较规则,将使用该表所在数据库的字符集和比较规则作 为该表的字符集和比较规则
5.2.4列级别
- 对于存储字符串的列,同一个表中的不同的列也可以有不同的字符集和比较规则。我们在创建和修改列定义的时候可以指定该列的字符集和比较规则,语法如下:
CREATE TABLE 表名( 列名 字符串类型 [CHARACTER SET 字符集名称] [COLLATE 比较规则名称], 其他列... ); ALTER TABLE 表名 MODIFY 列名 字符串类型 [CHARACTER SET 字符集名称] [COLLATE 比较规则名称];
- 对于某个列来说,如果在创建和修改的语句中没有指明字符集和比较规则,将使用该列所在表的字符集和比较规则作为该列的字符集和比较规则
- 在转换列的字符集时需要注意,如果转换前列中存储的数据不能用转换后的字符集进行表示会发生 错误。比方说原先列使用的字符集是utf8,列中存储了一些汉字,现在把列的字符集转换为ascii的 话就会出错,因为ascii字符集并不能表示汉字字符
(3)字符集与比较规则
- utf8与utf8mb4:utf8 字符集表示一个字符需要使用1~4个字节,但是我们常用的一些字符使用1~3个字节就可以表示了。而字符集表示一个字符所用的最大字节长度,在某些方面会影响系统的存储和性能,所以设计 MySQL的设计者偷偷的定义了两个概念
- utf8mb3 :阉割过的utf8 字符集,只使用1~3个字节表示字符
- utf8mb4 :正宗的 utf8 字符集,使用1~4个字节表示字符
- 比较规则:MySQL版本一共支持41种字符集,其中的Default collation列表示这种字符集中一种默认的比较规则,里面包含着该比较规则主要作用于哪种语言,比如 utf8_polish_ci表示以波兰语的规则 比较,utf8_spanish_ci是以西班牙语的规则比较, utf8_general_ci是一种通用的比较规则
- 后缀表示该比较规则是否区分语言中的重音、大小写。具体如下(最后一列 Maxlen ,它代表该种字符集表示一个字符最多需要几个字节):
后缀 英文释义 描述 _ai accent insensitive 不区分重音 _as accent sensitive 区分重音 _ci case insensitive 不区分大小写 _cs case sensitive 区分大小写 _bin binary 以二进制方式比较 - 常用操作一:
#查看GBK字符集的比较规则 SHOW COLLATION LIKE 'gbk%'; #查看UTF-8字符集的比较规则 SHOW COLLATION LIKE 'utf8%';
- 常用操作二:
#查看服务器的字符集和比较规则 SHOW VARIABLES LIKE '%_server'; #查看数据库的字符集和比较规则 SHOW VARIABLES LIKE '%_database'; #查看具体数据库的字符集 SHOW CREATE DATABASE dbtest1; #修改具体数据库的字符集 ALTER DATABASE dbtest1 DEFAULT CHARACTER SET 'utf8' COLLATE 'utf8_general_ci';
- 常用操作三:
#查看表的字符集 show create table employees; #查看表的比较规则 show table status from atguigudb like 'employees'; #修改表的字符集和比较规则 ALTER TABLE emp1 DEFAULT CHARACTER SET 'utf8' COLLATE 'utf8_general_ci'
(4)请求到响应过程中字符集的变化
- 说明:
系统变量 描述 character_set_client 服务器解码请求时使用的字符集 character_set_connection 服务器处理请求时会把请求字符串从 character_set_client 转为 character_set_connection character_set_results 服务器向客户端返回数据时使用的字符集 - 假设我们客户端发送的请求是“我”这个字符串,为了方便大家理解这个过程,我们只分析字符 '我' 在这个过程中字符集的转换
- 客户端发送请求所使用的字符集。一般情况下客户端所使用的字符集和当前操作系统一致,不同操作系统使用的字符集可能不一 样,如:(1)类 Unix 系统使用的是 utf8(2)Windows 使用的是 gbk。当客户端使用的是 utf8 字符集,字符 '我' 在发送给服务器的请求中的字节形式就是: 0xE68891。注意:如果你使用的是可视化工具,比如navicat之类的,这些工具可能会使用自定义的字符集来编码发送到服务器的字符串,而不采用操作系统默认的字符集(所以在学习的时候还是尽量用命令行窗口)
- 服务器接收到客户端发送来的请求其实是一串二进制的字节,它会认为这串字节采用的字符集是 character_set_client ,然后把这串字节转换为 character_set_connection 字符集编码的字符。由于我的计算机上 character_set_client 的值是 utf8 ,首先会按照 utf8 字符集对字节串 0xE68891 进行解码,得到的字符串就是 '我' ,然后按照 character_set_connection代表的字符集,也就是 gbk 进行编码,得到的结果就是字节串 0xCED2
- 因为表 t 的列 col 采用的是gbk字符集,与character_set_connection一致,所以直接到列中找字节值为0xCED2的记录,最后找到了一条记录。注意:如果某个列使用的字符集和character_set_connection代表的字符集不一致的话,还需要进行一次字符集转换
- 上一步骤找到的记录中的 col 列其实是一个字节串 0xCED2 , col 列是采用gbk进行编码的,所以首先会将这个字节串使用gbk进行解码,得到字符串'我' ,然后再把这个字符串使用character_set_results代表的字符集,也就是utf8进行编码,得到了新的字节串:0xE68891,然后发送给客户端
- 由于客户端是用的字符集是 utf8 ,所以可以顺利的将 0xE68891解释成字符“我”,从而显示到我们的显示器上,所以我们人类也读懂了返回的结果
- 如图:
六、SQL大小写规范
(1)Windows和Linux平台的区别
- 在 SQL 中,关键字和函数名是不用区分字母大小写的,比如 SELECT、WHERE、ORDER、GROUP BY 等关 键字,以及 ABS、MOD、ROUND、MAX 等函数名
- 不过在 SQL 中,你还是要确定大小写的规范,因为在 Linux 和 Windows 环境下,你可能会遇到不同的大小写问题。 Windows系统默认大小写不敏感 ,但是Linux系统是大小写敏感的
- 通过如下命令查看:
- 代码:
SHOW VARIABLES LIKE '%lower_case_table_names%;
- Windows系统下:
- Linux系统下:
- 代码:
- lower_case_table_names参数值的设置:
- 默认为0,大小写敏感
- 设置1,大小写不敏感。创建的表,数据库都是以小写形式存放在磁盘上,对于sql语句都是转换为小写对表和数据库进行查找
- 设置2,创建的表和数据库依据语句上格式存放,凡是查找都转换为小写进行
- 两个平台上SQL大小写的区别具体来说:
- 数据库名、表名、表的别名、变量名是严格区分大小写的
- 关键字、函数名称在SQL中不区分大小写
- 列名(或字段名)与列的别名(或字段的别名)在所有情况下均是忽略大小写的
- MySQL在Windows环境下全部不区分大小写
(2)Linux下大小写规则设置
- 当想设置为大小写不敏感时,要在my.cnf这个配置文件[mysqld]中加入 lower_case_table_names=1 ,然后重启服务器
- 但是在重启数据库实例之前,就需要将原来的数据库和表转换为小写,否则将找不到数据库名
- 此参数适用于MySQL5.7。在MySQL8.0下禁止在重新启动MySQL服务时将lower_case_table_names 设置成不同于初始化 MySQL 服务时设置的 lower_case_table_names 值。如果非要将MySQL8设置为大小写不敏感,具体步骤为:
- 停止MySQL服务
- 删除数据目录,即删除/var/lib/mysql目录
- 在MySQL配置文件( /etc/my.cnf )中添加 lower_case_table_names=1
- 启动MySQL服务
(3)SQL编写建议
- 如果你的变量名命名规范没有统一,就可能产生错误。这里有一个有关命名规范的建议:
- 关键字和函数名称全部大写
- 数据库名、表名、表别名、字段名、字段别名等全部小写
- SQL语句必须以分号结尾
- 数据库名、表名和变量名在 Linux MySQL 环境下是区分大小写的,因此建议你统一这些字段的命名规则,比如全部采用小写的方式
- 虽然关键字和函数名称在 SQL 中不区分大小写,也就是如果小写的话同样可以执行。但是同时将关键词和函数名称全部大写,以便于区分数据库名、表名、字段名
七、sql_mode的合理设置
(1)介绍
- sql_mode会影响MySQL支持的SQL语法以及它执行的数据验证检查。通过设置sql_mode,可以完成不同严格程度的数据校验,有效地保障数据的准确性
- MySQL服务器可以在不同的SQL模式下运行,并且可以针对不同的客户端以不同的方式应用这些模式,具体取决于sql_mode系统变量的值
- MySQL5.6和MySQL5.7默认的sql_mode模式参数是不一样的:
- 5.6的mode值默认为空(NO_ENGINE_SUBSTITUTION),其实表示的是一个空值,相当于没有什么模式设置,可以理解为宽松模式。在这种设置下,是可以允许一些非法操作的,比如允许一些非法数据的插入
- 5.7的mode是严格模式(STRICT_TRANS_TABLES)。用于进行数据的严格校验,错误数据不能插入,报error(错误),并且事务回滚
(2)宽松模式vs严格模式
7.2.1宽松模式
- 如果设置的是宽松模式,那么我们在插入数据的时候,即便是给了一个错误的数据,也可能会被接受, 并且不报错
- 举例:我在创建一个表时,该表中有一个字段为name,给name设置的字段类型时 char(10) ,如果我 在插入数据的时候,其中name这个字段对应的有一条数据的 长度超过了10 ,例如'1234567890abc',超 过了设定的字段长度10,那么不会报错,并且取前10个字符存上,也就是说你这个数据被存为 了'1234567890',而'abc'就没有了。但是,我们给的这条数据是错误的,因为超过了字段长度,但是并没 有报错,并且mysql自行处理并接受了,这就是宽松模式的效果
- 应用场景:通过设置sql mode为宽松模式,来保证大多数sql符合标准的sql语法,这样应用在不同数据库之间进行迁移时,则不需要对业务sql进行较大的修改
7.2.2严格模式
- 出现上面宽松模式的错误,应该报错才对,所以MySQL5.7版本就将sql_mode默认值改为了严格模式。所以在生产等环境中,我们必须采用的是严格模式,进而开发、测试环境的数据库也必须要设置,这样在开发测试阶段就可以发现问题。并且我们即便是用的MySQL5.6,也应该自行将其改为严格模式
- 改为严格模式后可能存在的问题:若设置模式中包含了NO_ZERO_DATE,那么MySQL数据库不允许插入零日期,插入零日期会抛出错误而不是警告。例如,表中含字段TIMESTAMP列(如果未声明为NULL或显示DEFAULT子句)将自动分配 DEFAULT '0000-00-00 00:00:00'(零时间戳),这显然是不满足sql_mode中的NO_ZERO_DATE而报错
(3)宽松模式再举例
(4)模式查看和设置
- 查看当前的sql_mode:
select @@session.sql_mode; select @@global.sql_mode; #或者 show variables like 'sql_mode';
- 临时设置方式:
SET GLOBAL sql_mode = 'modes...'; #全局 SET SESSION sql_mode = 'modes...'; #当前会话
- 永久设置方式:在/etc/my.cnf中配置sql_mode,即在my.cnf文件(windows系统是my.ini文件),新增(别忘了要重启MySQL服务):
[mysqld] 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服务的,所以采用临时设置方式 + 永久设置方式来解决线上的问题, 那么即便是有一天真的重启了MySQL服务,也会永久生效了
(5)sql_mode常用值
- sql_mode常用值:
参数值 说明 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后,不能用双引号来引用字符串,因为它被解释为识别符