在Docker中,安装两个版本的Mysql,一个5.7的,一个8.0的,因为我的虚机上已经安装了5.7的。因此这里再用Docker安装下8.0的。
用Docker安装Mysql8.0
1.首先先看下Docker基本的命令,看下Docker中有哪些镜像:
docker images
结果如下:
2.安装8.0版本的Mysql:格式:docker pull [镜像名称]:[版本]
docker pull mysql:8.0
3.用该镜像启动一个容器:
sudo docker run -p 3306:3306 --name mysql8.0 \
-v /mydata/mysql8.0/log:/var/log/mysql \
-v /mydata/mysql8.0/data:/var/lib/mysql \
-v /mydata/mysql8.0/conf:/etc/mysql \
-v /mydata/lig/mysql-files:/var/lib/mysql-files \
-e MYSQL_ROOT_PASSWORD=你的Mysql密码 \
-d mysql:8.0
4.查看容器是否启动成功:
docker ps
结果如下:
注意:
安装Mysql8.0和5.7有个细微的区别,若启动的时候指定了外部配置存储,需要比Mysql5.7多一个配置,否则会报错。
-v /mydata/lig/mysql-files:/var/lib/mysql-files \
在mysql8.0开始,数据库的默认字符集改为了utf8
。而Mysql5.7这类的默认字符集都是latin1
。
查看字符集命令:
show VARIABLES like '%character%'
create DATABASE dbtest1;
use dbtest1;
create table user(id int,name varchar(20));
insert into user(id,name) values(1,'Tom');
insert into user(id,name) values(1,'林家俊');
select * from user;
结果如下:
倘若我在Mysql5.7版本中再重复一遍上述的过程,看看结果发生了报错:
因为我们此时添加了中文数据,但是呢,我们表和数据库使用的字符集是latin1
,是不支持中文的。 比如查看字符集:
show create DATABASE dbtest1
因为我的Mysql使用的是外部配置映射,根据上述的配置信息(Mysql8.0和5.7的配置路径是一样的,只不过机器不一样),前往/mydata/mysql/conf
,添加my.cnf
文件:
[mysqld]
character_set_server=utf8
然后重启mysql: docker restart mysql
,此时我们再看下字符集
可以发现,我们的默认字符集已经改成了utf8
,但是我们已经创建的表和数据库的字符集是不能够被顺带修改的。在今后新创建的表或者数据库的时候,才会应用这个配置。 例如我们此时创建个新的数据库dbtest2
:
create DATABASE dbtest2;
use dbtest2;
create table user(id int,name varchar(20));
insert into user(id,name) values(1,'Tom');
insert into user(id,name) values(1,'林家俊');
select * from user;
use dbtest1;
# 修改数据库的字符集
ALTER DATABASE dbtest1 character set 'utf8';
# 修改表的字符集
ALTER TABLE user convert to CHARACTER set 'utf8';
Mysql字符集的级别分为4种:层级关系由上到下递增。
服务器级别字符集设置,一般我们通过配置文件来进行,例如:
[mysqld]
character_set_server=utf8
数据库级别字符集设置,具体语法如:
create database [数据库名] character set [字符集名称];
alter database [数据库名] character set [字符集名称];
表级别字符集设置,具体语法如:
create table [表名] character set [字符集名称];
alter table [表名] character set [字符集名称];
列级别字符集设置,具体语法如:
create table [表名](
[列名] [字符串类型] [character set xxx],
...,
);
utf8
和utf8mb4
的区别:(我们可以发现Mysql8.0默认字符集为utf8mb4
)
utf8
:表示一个字符需要使用1~4个字节表示字符。utf8mb3
:缩略版的utf8
字符集,只使用1~3个字节表示字符。utf8mb4
:正宗的utf8
字符集。使用1~4个字节表示字符。注意:Mysql中utf8
是utf8mb3
的别名。若有需求去存储4字节编码一个字符的情况,例如emoji表情,那就需要设置字符集为utf8mb4
.
证明:show CHARSET
我们可以发现第三列Default collation
,代表字符集的默认比较规则。
后缀 | 描述 |
---|---|
_ai | 不区分重音 |
_as | 区分重音 |
_ci | 不区分大小写 |
_cs | 区分大小写 |
_bin | 以二进制方式比较 |
重要的几个点:
utf8_general_ci
和utf8_unicode_ci
对于中英文来说没有什么实质区别。utf8_general_ci
相对而言速度快,但是准确度较差。utf8_unicode_ci
准确度高,但是速度较慢。并且适用于多语言的比较。查看和修改表的比较规则:
show table status from dbtest1 like '%user';
ALTER table user DEFAULT CHARACTER SET 'utf8' COLLATE 'utf8_general_ci';
我们先来看下3种字符集的系统变量:
character_set_client
:服务器解码的时候使用的字符集。character_set_connection
:服务器处理请求的时候会将character_set_client
字符集转化为character_set_connection
字符集。(相当于传输过程中的一个中间态)character_set_results
:服务器向客户端返回数据的时候使用的字符集。从流程当中我们可以注意以下几点:
character_set_client
、character_set_results
字符集必须三者一致。不然进行请求的时候,服务器可能无法识别对应的字符。character_set_results
不一致,会导致客户端无法解码。也就是所谓的乱码现象(本质原因了)。utf8mb4
。大家可以用该命令进行查看:
show VARIABLES like '%lower_case_table_names%'
Mysql对于linux和window环境下的大小写规范区别如下:
我们知道,Mysql的存储引擎我们较为熟知的有InnoDB与MyISAM。这样的存储引擎都是将表存储在磁盘上的。我们将操作系统用来管理磁盘的结构称之为文件系统。
这个命令也比较简单:
show DATABASES;
mysql
:核心数据库,存储了Mysql的用户和权限信息、存储过程、事件的定义等。information_schema
:保存着Mysql服务器维护的其他数据库的信息,例如有哪些表、视图、触发器、索引等等。performance_schema
:保存Mysql服务器运行过程中的状态信息,一般用于监控Mysql的各类性能指标,比如最近执行了哪些语句。执行过程花了多少时间等。sys
:主要通过视图的方式把information_schema
数据库和performance_schema
数据库结合起来。以本篇文章创建的dbtest1
数据库为例,我们去/mydata/mysql/data/dbtest1
目录下去查看数据(我这里做了外部文件映射,不然就是/var/lib/mysql/data/
)
db.opt
:存储了dbtest1
这个数据库的一些基本信息,如使用的字符集、比较规则等。user.frm
:存储了user
这张表的表结构。user.ibd
:存放user
这张表中的数据(独立表空间)。ibdata1
、大小为12M的文件。xxx.ibd
文件。[server]
# 0:系统表空间,1:独立表空间
innodb_file_per_table=0
小结下就是:若表使用InnoDB存储引擎,一张表会产生1~2个文件:
xxx.frm
:描述表结构文件。ibdata1
文件中。xxx.idb
文件用于存储数据和索引信息。xxx.frm
文件,而是将其整合到xxx.idb
文件中了。若采用MyISAM存储引擎:一张表会产生3种文件
xxx.frm
文件。Mysql8.0中:xxx.sdi
文件。两者都是用来描述表结构和字段长度的。xxx.MYD
文件:数据信息文件。xxx.MYI
文件:存储索引信息文件。在Mysql8.0和5.7中分别创建表:
use dbtest1;
CREATE TABLE `student` (
`id` bigint(20) DEFAULT NULL,
`name` varchar(64) DEFAULT NULL
) ENGINE=MyISAM DEFAULT CHARSET=utf8;
Mysql5.7中的文件类型:
Mysql8.0中的文件类型:
Mysql的登录命令如下:
mysql -h hostname -p port -u username -p databaseName -e "SQL语句"
用户的权限查看:
show PRIVILEGES;
授予权限原则:
授权命令:
grant [权限1],[权限2] on [数据库名称 | *].[表名 | *] to [用户名]@[用户地址]
例如:
我在Mysql8.0中创建一个用户
create user 'ljj' IDENTIFIED by '000000';
# 赋予这个用户对数据库dbtest1下的student表,查询和插入的权限
GRANT SELECT ,INSERT on dbtest1.student to 'ljj'@'%';
此时我用ljj
用户来访问Docker中的Mysql:
1.先查看对应容器的id: docker ps
2.复制对应的容器id,执行命令:docker exec -it [你的容器id] bash
,进入容器内部。
3.连接Mysql客户端:mysql -u ljj --p[密码]
4.测试权限:
use dbtest1;
INSERT into student(id,name) VALUE(1,'3');
SELECT * from student;
UPDATE student SET name ='3' WHERE id = 1;
倘若某个用户被删除,那么我们需要回收对应的权限,命令如下:
revoke [权限1],[权限2] on [数据库名称 | *].[表名 | *] from [用户名]@[用户地址]
用户在操作Mysql的时候,Mysql首先会核实该用户对应的操作请求是否被允许,而这个过程叫做访问控制过程。分为两个阶段:
连接核实阶段:
user
表中的host、user、authentication_string
这三个字段来匹配客户端提供的信息。请求核实阶段:
user
表,若指定的权限没有在user
表中被授予,此时检查db
表。db
表中的权限限制于数据库层级。该层级中的SELECT
权限允许用户查看置顶数据库的所有表数据。db
表中依旧没有找到权限,则检查tables_priv
表以及columns_priv
表。客户端发送一个请求的时候,Mysql会先去缓存中去寻找这条SQL语句。
值得注意的是:查询缓存的效率并不高,因此Mysql8.0之后抛弃了这个功能。
缓存的形式:Key-Value
。
Key
:执行过的语句。Value
:对应语句的结果、那么为什么查询缓存的效率不高呢?原因有三点:
第一点:只有相同的查询操作才会命中查询缓存。若两个查询请求有任何字符上的不同(空格、字符、大小写等),都会导致缓存不会命中,因此Mysql的查询缓存命中率并不高。
例如:
select * from user;
# 这里多了几个空格,那么这两个查询语句作为key明显不一样,长度都不一样
select * from user;
第二点:若查询请求中包含某些系统函数、用户自定义变量和函数或者查询系统表。那么该请求不会被缓存。
例如函数NOW
,每次调用都会产生不同的结果,那么这样的SQL语句是不会被加入到查询缓存当中的。
第三点:缓存有失效时间,Mysql的缓存系统会涉及到每张表。但是只要该表的结构或者数据被修改(Insert、Update、Delete)等操作,那么该表相关的高速缓存查询都会置为无效并从缓存中删除。
因此对于更新操作频繁的数据库来说, 查询缓存的命中率低的不行。
那么什么样的情况适合使用查询缓存呢?
如果我们有一些表,基本上不会涉及到更新,比如系统配置表、字典表等。那么我们可以在这张表的查询上使用查询缓存。
如何开启?在my.cnf配置文件中添加:
query_cache_type=2
query_cache_type
有3个值:
SQL_CACHE
关键词的时候才缓存。例如我希望对以下SQL进行缓存:
select SQL_CACHE * from config where id = 3;
查看当前数据库是否使用查询缓存:
show VARIABLES like '%query_cache_type%'
监控查询缓存的命中情况:
show status like '%Qcache%'
结果如下:
Qcache_free_blocks
:空闲的block
数量,数值越大,代表缓存中的碎片越多。
Qcache_free_memory
:缓存大小。
Qcache_lowmem_prunes
:有多少条缓存是因为内存不够而命中不到。若值比较大,说明需要增加查询缓存的内存大小。
Qcache_not_cached
:表示因为query_cache_type
参数的设置而没有命中查询缓存的次数。
Qcache_hits
:表示有多少次命中缓存。数字越大,效果越理想。
Qcache_inserts
:表示多少次未命中缓存然后将结果插入到缓存中的情况。
在查询缓存阶段结束后,就该进入解析器解析阶段了。
这一阶段,Mysql的目的是需要知道传入的SQL语句是要做什么事情,因此需要对其做词法、语法的分析。
词法分析中:
select
关键字,那么此时该SQL是一条查询语句。xxx
识别成表名xxx
,找到对应的表。id
的识别等。语法分析中:
该阶段主要是是确定SQL语句的执行方式, 比如:是全表搜索,还是索引检索。
一条查询语句可以有很多种执行方式,最后都返回相同的结果,而优化器的作用就是找到最优的执行方案。
例如:
而优化又分为两个阶段:
到这里为止,Mysql服务器已经有了一个执行计划,准备交给执行器来执行。
本阶段主要是调用存储引擎的API对表进行读写操作(在有权限的前提下)
总结下执行流程就是:
此时我们如果打开Mysql5.7的查询缓存功能(8.0该功能已抛弃),在配置文件my.cnf
中添加属性:
query_cache_type=1
然后重启docker:
docker restart mysql
此时在查看查询缓存的启用情况:
确认profiling
是否开启,开启它后可以让Mysql手机在SQL执行时所使用的资源情况。
SELECT @@profiling;
若你的值为0,代表关闭,可以使用命令将其临时开启:
同时在设置以下profiling
:
# 这种设置方式是临时的
set @@profiling=1;
同样我们执行两边相同的SQL语句,例如:
select name from user where id =1
然后查看SQL执行过程:
show PROFILES;
此时我们查看下详情
# 默认是最近的一次查询
show PROFILE;
# 指定查询语句ID查看过程
show PROFILE for query 4;