MySQL之存储引擎及SQL优化

6、Mysql的体系结构概念

在这里插入图片描述

整个MySQL Server由以下组成

  • Connection Pool:连接组件(接受客户端请求,进行认证授权后处理)
  • Management Services & Utilltles:管理服务和工具组件(备份、恢复,sql语句的封装 优化 缓存处理)
  • SQL interface:SQL接口组件
  • Parser:查询分析器组件
  • Caches & Buffers:缓冲池组件
  • Pluggable Storage Engines:存储引擎
  • File System:文件系统
  1. 连接层

​ 最上层是一些客户端和链接服务,包含本地sock 通信和大多数基于客户端/服务端工具实现的类似于 TCP/IP的通 信。主要完成一些类似于连接处理、授权认证、及相关的安全方案。在该层上引入了线程池的概念,为通过认证安 全接入的客户端提供线程。同样在该层上可以实现基于SSL的安全链接。服务器也会为安全接入的每个客户端验证 它所具有的操作权限。

  1. 服务层

​ 第二层架构主要完成大多数的核心服务功能,如SQL接口,并完成缓存的查询,SQL的分析和优化,部分内置函数的执行。所有跨存储引擎的功能也在这一层实现,如 过程、函数等。在该层,服务器会解析查询并创建相应的内部 解析树,并对其完成相应的优化如确定表的查询的顺序,是否利用索引等, 最后生成相应的执行操作。如果是 select语句,服务器还会查询内部的缓存,如果缓存空间足够大,这样在解决大量读操作的环境中能够很好的提升 系统的性能。

  1. 引擎层

     存储引擎层, 存储引擎真正的负责了MySQL中数据的存储和提取,服务器通过API和存储引擎进行通信。不同的存 
    

​ 储引擎具有不同的功能,这样我们可以根据自己的需要,来选取合适的存储引擎。 4)存储层
数据存储层, 主要是将数据存储在文件系统之上,并完成与存储引擎的交互。

​ 和其他数据库相比,MySQL有点与众不同,它的架构可以在多种不同场景中应用并发挥良好作用。主要体现在存储 引擎上,插件式的存储引擎架构,将查询处理和其他的系统任务以及数据的存储提取分离。这种架构可以根据业务 的需求和实际需要选择合适的存储引擎

7、存储引擎

7.1 存储引擎概述

​ 和大多数的数据库不同, MySQL中有一个存储引擎的概念, 针对不同的存储需求可以选择最优的存储引擎。 存储引擎就是存储数据,建立索引,更新查询数据等等技术的实现方式 。存储引擎是基于表的,而不是基于库的。

所以存储引擎也可被称为表类型。

​ Oracle,SqlServer等数据库只有一种存储引擎。MySQL提供了插件式的存储引擎架构。所以MySQL存在多种存储 引擎,可以根据需要使用相应引擎,或者编写存储引擎。

​ MySQL5.0支持的存储引擎包含 : InnoDB 、MyISAM 、BDB、MEMORY、MERGE、EXAMPLE、NDB Cluster、 ARCHIVE、CSV、BLACKHOLE、FEDERATED等,其中InnoDB和BDB提供事务安全表,其他存储引擎是非事务安 全表。

可以通过指定 show engines , 来查询当前数据库支持的存储引擎 :

-- 查询当前数据库支持的存储引擎
mysql> show engines;
-- 查看默认的存储引擎
mysql> show variables like '%storage_engine%';
+----------------------------------+--------+
| Variable_name                    | Value  |
+----------------------------------+--------+
| default_storage_engine           | InnoDB |
| default_tmp_storage_engine       | InnoDB |
| disabled_storage_engines         |        |
| internal_tmp_disk_storage_engine | InnoDB |
+----------------------------------+--------+

7.2 各种存储引擎特征

下面重点介绍几种常用的存储引擎, 并对比各个存储引擎之间的区别, 如下表所示 :

特点InnoDBMyISAMMEMORYMERGENDB
存储限制64TB没有
事务安全支持
锁机制行锁(适合高并发)表锁表锁表锁行锁
B树索引支持支持支持支持支持
哈希索引支持
全文索引支持(5.6版本后)支持
集群索引支持
数据索引支持
索引缓存支持支持支持支持支持
数据可压缩支持
空间使用N/A
内存使用中等
批量插入速度
支持外键支持

下面我们将重点介绍最长使用的两种存储引擎: InnoDB、MyISAM , 另外两种 MEMORY、MERGE , 了解即
可。

7.2.1 InnoDB

​ InnoDB存储引擎是Mysql的默认存储引擎。InnoDB存储引擎提供了具有提交、回滚、崩溃恢复能力的事务安全。

但是对比MyISAM的存储引擎,InnoDB写的处理效率差一些,并且会占用更多的磁盘空间以保留数据和索引。

InnoDB存储引擎不同于其他存储引擎的特点 :

  • 事务控制:
create table goods_innodb(
	id int NOT NULL AUTO_INCREMENT, 
  name varchar(20) NOT NULL, 
  primary key(id)
)ENGINE=innodb DEFAULT CHARSET=utf8;

-- 会话一
mysql> start transaction;
Query OK, 0 rows affected (0.00 sec)

mysql> insert into goods_innodb(id,name)values(null,'Meta20');
Query OK, 1 row affected (0.00 sec)

mysql> commit;
Query OK, 0 rows affected (0.00 sec)
-- 会话二
mysql> select * from goods_innodb;
Empty set (0.01 sec)

mysql> select * from goods_innodb;
+----+--------+
| id | name   |
+----+--------+
|  1 | Meta20 |
+----+--------+
1 row in set (0.00 sec)

-- 总结
-- 当会话一的事务提交后,会话二才能查询出会话一insert进去的数据
  • 外键约束

    ​ MySQL支持外键的存储引擎只有InnoDB , 在创建外键的时候, 要求父表必须有对应的索引 , 子表在创建外键的时候, 也会自动的创建对应的索引。

create table country_innodb(
		country_id int NOT NULL AUTO_INCREMENT, 
  	country_name varchar(100) NOT NULL, 
		primary key(country_id)
)ENGINE=InnoDB DEFAULT CHARSET=utf8;
create table city_innodb(
    city_id int NOT NULL AUTO_INCREMENT,
    city_name varchar(50) NOT NULL,
    country_id int NOT NULL,
    primary key(city_id),
    key idx_fk_country_id(country_id),
    CONSTRAINT `fk_city_country` FOREIGN KEY(country_id) REFERENCES
country_innodb(country_id) ON DELETE RESTRICT ON UPDATE CASCADE
)ENGINE=InnoDB DEFAULT CHARSET=utf8;

insert into country_innodb values(null,'China'),(null,'America'),(null,'Japan');
insert into city_innodb values(null,'Xian',1),(null,'NewYork',2),(null,'BeiJing',1);


-- ON DELETE RESTRICT 删除主表数据时,如果有关联记录,不删除;
-- ON UPDATE CASCADE 更新主表时,如果子表有关联记录,更新子表字段

在创建表时,可以指定在删除、更新父表时,对子表进行相应的操作,包括RESTRICT、CASCADE、SET NULL和NO ACTION

  • RESTRICT和NO ACTION相同,表示:限制在子表有关联记录的情况下,父表不能更新;(子表有关联记录,父表不能更新)

  • CASCADE表示:父表在更新或者删除时,更新或者删除子表对应的记录;(父表和子表同步更新或者删除)

  • SET NULL则表示:父表在更新或者删除的时候,子表对应字段被SET NULL

针对上面创建的俩张表,子表的外键指定是ON DELETE RESTRICT ON UPDATE CASCADE

mysql> select * from city_innodb;
+---------+-----------+------------+
| city_id | city_name | country_id |
+---------+-----------+------------+
|       1 | Xian      |          1 |
|       2 | NewYork   |          2 |
|       3 | BeiJing   |          1 |
+---------+-----------+------------+
3 rows in set (0.00 sec)

mysql> select * from country_innodb;
+------------+--------------+
| country_id | country_name |
+------------+--------------+
|          1 | China        |
|          2 | America      |
|          3 | Japan        |
+------------+--------------+
3 rows in set (0.00 sec)

mysql> delect from city_innodb where country_id =1;
ERROR 1064 (42000): You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near 'delect from city_innodb where country_id =1' at line 1

mysql> update country_innodb set country_id = 100 where country_id =1;
Query OK, 1 row affected (0.00 sec)
Rows matched: 1  Changed: 1  Warnings: 0

mysql> select * from city_innodb;
+---------+-----------+------------+
| city_id | city_name | country_id |
+---------+-----------+------------+
|       1 | Xian      |        100 |
|       2 | NewYork   |          2 |
|       3 | BeiJing   |        100 |
+---------+-----------+------------+
3 rows in set (0.00 sec)

存储方式

InnoDB存储表和索引有以下俩种方式:

  • 1、使用共享表空间存储,这种方式创建的表的表结构保存在.frm文件中,数据和索引保存在innodb_data_home_dir和innodb_data_file_path定义的表空间中,可以是多个文件。
  • 2、使用多表空间存储,这种方式创建的表的表结构仍然存在.frm文件中,但是每个表的数据和索引单独保存在.ibd中
7.2.2 MyISAM

​ MyISAM不支持事务、也不支持外键,其优势是访问的速度快,对事务的完整性没有要求或者以SELECT、INSERT为主的应用基础上都可以使用这个引擎来创建表,有以下俩个比较重要的特点:

不支持事务

create table goods_myisam(
id int NOT NULL AUTO_INCREMENT, name varchar(20) NOT NULL, primary key(id)
)ENGINE=myisam DEFAULT CHARSET=utf8;

-- 会话一
mysql> start transaction;
Query OK, 0 rows affected (0.00 sec)

mysql> insert into goods_myisam(id,name) values(null,'mysql');
Query OK, 1 row affected (0.01 sec)

-- 会话二
mysql> select * from goods_myisam;
+----+-------+
| id | name  |
+----+-------+
|  1 | mysql |
+----+-------+
1 row in set (0.00 sec)

-- 会话一
mysql> rollback;
Query OK, 0 rows affected, 1 warning (0.00 sec)

-- 回话二
mysql> select * from goods_myisam;
+----+-------+
| id | name  |
+----+-------+
|  1 | mysql |
+----+-------+
1 row in set (0.00 sec)
7.2.3 MEMORY

​ Memory存储引擎将表的数据存放在内存中。每个MEMORY表实际对应一个磁盘文件,格式是.frm ,该文件中只 存储表的结构,而其数据文件,都是存储在内存中,这样有利于数据的快速处理,提高整个表的效率。MEMORY 类型的表访问非常地快,因为他的数据是存放在内存中的,并且默认使用HASH索引 , 但是服务一旦关闭,表中的 数据就会丢失。

7.2.4 MERGE

​ MERGE存储引擎是一组MyISAM表的组合,这些MyISAM表必须结构完全相同,MERGE表本身并没有存储数据,对

MERGE类型的表可以进行查询、更新、删除操作,这些操作实际上是对内部的MyISAM表进行的。

对于MERGE类型表的插入操作,是通过INSERT_METHOD子句定义插入的表,可以有3个不同的值,使用FIRST 或 LAST 值使得插入操作被相应地作用在第一或者最后一个表上,不定义这个子句或者定义为NO,表示不能对这个 MERGE表执行插入操作。

可以对MERGE表进行DROP操作,但是这个操作只是删除MERGE表的定义,对内部的表是没有任何影响的。

7.3 存储引擎的选择

​ 在选择存储引擎时,应该根据应用系统的特点选择合适的存储引擎。对于复杂的应用系统,还可以根据实际情况选择多种存储引擎进行组合。以下是几种常用的存储引擎的使用环境。

  • InnoDB : 是Mysql的默认存储引擎,用于事务处理应用程序,支持外键。如果应用对事务的完整性有比较高 的要求,在并发条件下要求数据的一致性,数据操作除了插入和查询意外,还包含很多的更新、删除操作, 那么InnoDB存储引擎是比较合适的选择。InnoDB存储引擎除了有效的降低由于删除和更新导致的锁定, 还 可以确保事务的完整提交和回滚,对于类似于计费系统或者财务系统等对数据准确性要求比较高的系统, InnoDB是最合适的选择。

  • MyISAM : 如果应用是以读操作和插入操作为主,只有很少的更新和删除操作,并且对事务的完整性、并发 性要求不是很高,那么选择这个存储引擎是非常合适的。

  • MEMORY:将所有数据保存在RAM中,在需要快速定位记录和其他类似数据环境下,可以提供几块的访问。MEMORY的缺陷就是对表的大小有限制,太大的表无法缓存在内存中,其次是要确保表的数据可以恢复,数 据库异常终止后表中的数据是可以恢复的。MEMORY表通常用于更新不太频繁的小表,用以快速得到访问结 果。

  • MERGE:用于将一系列等同的MyISAM表以逻辑方式组合在一起,并作为一个对象引用他们。MERGE表的优 点在于可以突破对单个MyISAM表的大小限制,并且通过将不同的表分布在多个磁盘上,可以有效的改善 MERGE表的访问效率。这对于存储诸如数据仓储等VLDB环境十分合适。

8、优化SQL步骤

8.1 查看SQL执行频率

​ MySQL 客户端连接成功后,通过 show [session|global] status 命令可以提供服务器状态信息。show
[session|global] status 可以根据需要加上参数“session”或者“global”来显示 session 级(当前连接)的计结果和
global 级(自数据库上次启动至今)的统计结果。如果不写,默认使用参数是“session”。

-- 查询当前连接的状态信息
show status like 'Com_______';
-- 查询全局的状态信息
show global status like 'Com_______';
+---------------+-------+
| Variable_name | Value |
+---------------+-------+
| Com_binlog    | 0     |
| Com_commit    | 0     |
| Com_delete    | 0     |
| Com_insert    | 1     |
| Com_repair    | 0     |
| Com_revoke    | 0     |
| Com_select    | 8     |
| Com_signal    | 0     |
| Com_update    | 0     |
| Com_xa_end    | 0     |
+---------------+-------+
10 rows in set (0.01 sec)
show global status like 'innodb_rows_%';
mysql> show global status like 'innodb_rows_%';
+----------------------+-------+
| Variable_name        | Value |
+----------------------+-------+
| Innodb_rows_deleted  | 0     |
| Innodb_rows_inserted | 0     |
| Innodb_rows_read     | 8     |
| Innodb_rows_updated  | 0     |
+----------------------+-------+
4 rows in set (0.00 sec)

Com_xxx 表示每个 xxx 语句执行的次数,我们通常比较关心的是以下几个统计参数。

参数含义
Com_select执行 select 操作的次数,一次查询只累加 1。
Com_insert执行 INSERT 操作的次数,对于批量插入的 INSERT 操作,只累加一次。
Com_update执行 UPDATE 操作的次数。
Com_delete执行 DELETE 操作的次数。
Innodb_rows_readselect 查询返回的行数。
Innodb_rows_inserted执行 INSERT 操作插入的行数。
Innodb_rows_updated执行 UPDATE 操作更新的行数。
Innodb_rows_deleted执行 DELETE 操作删除的行数。
Connections试图连接 MySQL 服务器的次数。
Uptime服务器工作时间。
Slow_queries慢查询的次数。

Com_*** : 这些参数对于所有存储引擎的表操作都会进行累计。
Innodb_*** : 这几个参数只是针对InnoDB 存储引擎的,累加的算法也略有不同。

8.2 定位低效率执行SQL

  • 慢查询日志 : 通过慢查询日志定位那些执行效率较低的 SQL 语句,用–log-slow-queries[=file_name]选项启
    动时,mysqld 写一个包含所有执行时间超过 long_query_time 秒的 SQL 语句的日志文件。具体可以查看本
    书第 26 章中日志管理的相关部分。
  • show processlist:慢查询日志在查询结束以后才纪录,所以在应用反映执行效率出现问题的时候查询慢查询
    日志并不能定位问题,可以使用show processlist命令查看当前MySQL在进行的线程,包括线程的状态、是否
    锁表等,可以实时地查看 SQL 的执行情况,同时对一些锁表操作进行优化。
mysql> show processlist;
+----+------+-----------+------+---------+------+----------+------------------+
| Id | User | Host      | db   | Command | Time | State    | Info             |
+----+------+-----------+------+---------+------+----------+------------------+
|  3 | root | localhost | test | Query   |    0 | starting | show processlist |
+----+------+-----------+------+---------+------+----------+------------------+
1 row in set (0.00 sec)

-- 1)id列:用户登陆mysql时,系统分配的"connection_id",可以使用函数connection_id()查看(select connection_id();)
-- 2) user列:显示当前用户。如果不是root,这个命令就只显示用户权限范围的sql语句
-- 3) host列,显示这个语句是从哪个ip的哪个端口上发的,可以用来跟踪出现问题语句的用户
-- 4) db列,显示这个进程目前连接的是哪个数据库
-- 5) command列,显示当前连接的执行的命令,一般取值为休眠(sleep),查询(query),连接 (connect)等
-- 6) time列,显示这个状态持续的时间,单位是秒
-- 7) state列,显示使用当前连接的sql语句的状态,很重要的列。state描述的是语句执行中的某一个状态。一 个sql语句,以查询为例,可能需要经过copying to tmp table、sorting result、sending data等状态 才可以完成
-- 8)info列,显示这个sql语句,是判断问题语句的一个重要依据

8.3 explain分析执行计划

​ 通过以上步骤查询效率低的SQL语句后,可以通过explain或者desc命令获取MYSQL如何执行SELECT语句的信息,包括在SELECT语句执行过程中表如何连接和连接的顺序

查询SQL语句的执行计划:

mysql> explain select * from city;
+----+-------------+-------+------------+------+---------------+------+---------+------+------+----------+-------+
| id | select_type | table | partitions | type | possible_keys | key  | key_len | ref  | rows | filtered | Extra |
+----+-------------+-------+------------+------+---------------+------+---------+------+------+----------+-------+
|  1 | SIMPLE      | city  | NULL       | ALL  | NULL          | NULL | NULL    | NULL |    4 |   100.00 | NULL  |
+----+-------------+-------+------------+------+---------------+------+---------+------+------+----------+-------+
1 row in set, 1 warning (0.00 sec)
字段含义
IDelect查询的序列号,是一组数字,表示的是查询中执行select子句或者是操作表的顺序。
select_type表示 SELECT 的类型,常见的取值有 SIMPLE(简单表,即不使用表连接或者子查询)、PRIMARY(主查询,即外层的查询)、UNION(UNION 中的第二个或者后面的查询语句)、SUBQUERY(子查询中的第一个 SELECT)等
table输出结果集的表
type表示表的连接类型,性能由好到差的连接类型为( system —> const -----> eq_ref ------> ref-------> ref_or_null----> index_merge —> index_subquery -----> range -----> index ------>all )
possible_keys表示查询时,可能使用的索引
key表示实际使用的索引
key_len索引字段的长度
rows描行的数量
extra执行情况的说明和描述
8.3.1 环境准备
CREATE TABLE `t_role` (
	`id` varchar(32) NOT NULL,
	`role_name` varchar(255) DEFAULT NULL, 
  `role_code` varchar(255) DEFAULT NULL, 
  `description` varchar(255) DEFAULT NULL, 
  PRIMARY KEY (`id`),
	UNIQUE KEY `unique_role_name` (`role_name`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;

CREATE TABLE `t_user` (
  `id` varchar(32) NOT NULL,
  `username` varchar(45) NOT NULL,
  `password` varchar(96) NOT NULL,
  `name` varchar(45) NOT NULL,
  PRIMARY KEY (`id`),
  UNIQUE KEY `unique_user_username` (`username`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;

CREATE TABLE `user_role` (
  `id` int(11) NOT NULL auto_increment ,
  `user_id` varchar(32) DEFAULT NULL,
  `role_id` varchar(32) DEFAULT NULL,
  PRIMARY KEY (`id`),
  KEY `fk_ur_user_id` (`user_id`),
  KEY `fk_ur_role_id` (`role_id`),
  CONSTRAINT `fk_ur_role_id` FOREIGN KEY (`role_id`) REFERENCES `t_role` (`id`) ON
DELETE NO ACTION ON UPDATE NO ACTION,
  CONSTRAINT `fk_ur_user_id` FOREIGN KEY (`user_id`) REFERENCES `t_user` (`id`) ON
DELETE NO ACTION ON UPDATE NO ACTION
) ENGINE=InnoDB DEFAULT CHARSET=utf8;

insert into `t_user` (`id`, `username`, `password`, `name`) values('1','super','$2a$10$TJ4TmCdK.X4wv/tCqHW14.w70U3CC33CeVncD3SLmyMXMknstqKRe',' 超级管理员');
insert into `t_user` (`id`, `username`, `password`, `name`) values('2','admin','$2a$10$TJ4TmCdK.X4wv/tCqHW14.w70U3CC33CeVncD3SLmyMXMknstqKRe',' 系统管理员');
insert into `t_user` (`id`, `username`, `password`, `name`) values('3','itcast','$2a$10$8qmaHgUFUAmPR5pOuWhYWOr291WJYjHelUlYn07k5ELF8ZCrW0Cui', 'test02');
insert into `t_user` (`id`, `username`, `password`, `name`) values('4','stu1','$2a$10$pLtt2KDAFpwTWLjNsmTEi.oU1yOZyIn9XkziK/y/spH5rftCpUMZa','学 生1');
insert into `t_user` (`id`, `username`, `password`, `name`) values('5','stu2','$2a$10$nxPKkYSez7uz2YQYUnwhR.z57km3yqKn3Hr/p1FR6ZKgc18u.Tvqm','学 生2');
insert into `t_user` (`id`, `username`, `password`, `name`) values('6','t1','$2a$10$TJ4TmCdK.X4wv/tCqHW14.w70U3CC33CeVncD3SLmyMXMknstqKRe','老师 1');

INSERT INTO `t_role` (`id`, `role_name`, `role_code`, `description`) VALUES('5','学生','student','学生');
INSERT INTO `t_role` (`id`, `role_name`, `role_code`, `description`) VALUES('7','老师','teacher','老师');
INSERT INTO `t_role` (`id`, `role_name`, `role_code`, `description`) VALUES('8','教学管理员','teachmanager','教学管理员');
INSERT INTO `t_role` (`id`, `role_name`, `role_code`, `description`) VALUES('9','管理员','admin','管理员');
INSERT INTO `t_role` (`id`, `role_name`, `role_code`, `description`) VALUES('10','超级管理员','super','超级管理员');

INSERT INTO user_role(id,user_id,role_id) VALUES(NULL, '1', '5'),(NULL, '1', '7'),
(NULL, '2', '8'),(NULL, '3', '9'),(NULL, '4', '8'),(NULL, '5', '10');
8.3.2 explain 之id

id字段是select查询的序列号,是一组数字,表示的是查询中执行select子句或者是操作表的顺序、id情况分三种:

  • 1)id相同表示加载表的顺序是从上到下:
mysql> explain select * from t_role r, t_user u, user_role ur where r.id = ur.role_id and u.id = ur.user_id ;

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-tVL88IM7-1606377489294)(/Users/mac/Documents/1expalin分析执行计划之id1.png)]

  • 2)id不同id值越大,优先级越高,越先被执行。
EXPLAIN SELECT * FROM t_role WHERE id = (SELECT role_id FROM user_role WHERE user_id = (SELECT id FROM t_user WHERE username = 'stu1'));

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-boioCiAJ-1606377489297)(/Users/mac/Documents/1explain分析执行计划之id2.png)]

  • 3)id有相同,也有不同,同时存在。id相同的可以认为是一组,从上往下顺序执行;在所有的组中,id的值越大,优先级越高,越先执行。
EXPLAIN SELECT * FROM t_role r , (SELECT * FROM user_role ur WHERE ur.`user_id` = '2') a WHERE r.id = a.role_id ;
8.3.3 explain之select_type

表示select的类型,常见的取值如下:

select_type含义
SIMPLE简单的select查询,查询中不包含子查询或者UNION
PRIMARY查询中若包含任何复杂的子查询,最外层查询标记为该标识
SUBQUERY在SELECT或HWERE列表中包含子查询
DERIVED在FROM列表中包含的子查询,被标记为DERIVED(衍生) MYSQL会递归执行这些子查询,把结果放在临时表中(即:from后面接子查询,子查询结果被放入临时表)
UNION若第二个SELECT出现在UNION之后,则标记为UNION;若UNION包含在FROM子句的子查询中,外层SELECT被标记为:DERIVED
UNION RESULT从UNION表获取结果的SELECT

从SIMPLE到UNION RESULT 效率越来越低

8.3.4 explain之table

​ 表示查询的数据来源于哪张表

8.3.5 explain之type
type含义
NULLMYSQL不妨问任何表,索引直接返回结果
system表只有一行记录(等于系统表),这是const类型的特例,一般不会出现
const表示通过索引一次就找到了,const用于比较primary key或者unique索引,因为只匹配一行数据,所以很快,如将主键置于where列表中,mysql就能将该查询转换为一个常量,const于将"主键"或"唯一"的索引部分与常量进行比较 (即:根据主键或者唯一索引进行查询,且数据只有一条
eq_ref类似于ref,区别在于使用的是唯一索引,使用主键的关联查询,关联查询出的记录只有一条,常见于主键或唯一索引扫描(即:多表关联查询,查询结果只有一条记录
ref非唯一索引扫描,返回匹配某个单独值的所有行,本质上也是一种索引访问,返回多个匹配某个单独值的所有行(多个)(即:根据非唯一索引查询得到多条记录
range检索指定范围的行,where后面是一个范围查询(between、in、>、<、>=、特殊:in有时候会失效,从而转为 无索引 all)
indexindex与all的区别为index类型只是遍历了索引树,通常比all快,all是遍历数据文件
all将遍历全表找到匹配的行

结果值从最高到最坏依次为:

null > system > const > eq_ref > ref > fulltext > ref_or_null > index_range > unique_suquery > index_subquery > range > index > all 

system > const > eq_ref >ref > range > index > all

一般来说,我们需要保证查询至少达到range级别,最好达到ref

8.3.6 explain 之 key
possible_keys : 显示可能应用到这张表的索引,一个或多个

key : 实际用到的索引,如果为null,则没有使用索引

key_len : 表示索引中使用的字节数,该之为索引字段最大可能长度,并非实际使用长度,在不损失精确性的前提下,长度越短越好
8.3.7 explain 之 rows

​ 扫描行的数量

8.3.8 explain 之extra

其他额外的执行计划信息,在该列展示。

extra含义
using filesort说明mysql会对数据使用一个外部的索引排序,而不是按照表内的索引进行读取,称为"文件排序",效率低
using temporary使用了临时表保存中间结果,mysql在对查询结果排序的时候使用了临时表,常用于order by 和group by;效率低
using index表示相应的select操作使用了索引覆盖,避免访问表的数据行,效率不错
using where表示优化器需要通过索引回表查询数据
using index condition使用了索引但是需要回表查询数据
using index;using where查找使用了索引,但是需要的数据都能在索引列中找到,不需要回表查询数据。
impossible wherewhere子句永远为false

using index & using where 和using index区别在于是否用了where

  • using filesort
create table test02(
	a1 char(3),
  a2 char(3),
  a3 char(3),
  index index_a1(a1),
  index index_a2(a2),
  index index_a3(a3),
);
-- 单索引
explain select * from test02 where a1='' order by a1;
-- 出现 using filesort
explain select * from test02 where a1='' order by a2;
-- 小结:
-- 对于单索引,如果排序和查找是同一个字段,则不会出现using filesort;如果排序和查找不是同一个字段则会出现using filesort;
-- 避免: where哪些字段,就order by 哪些字段

-- 复合索引:不能跨列(最佳左前缀)
drop index index_a1 on test02;
drop index index_a2 on test02;
drop index index_a3 on test02;
alter table test02 add index index_a1_a2_a3(a1,a2,a3);

-- 出现 using filesort 
explain select * from test02 where a1='' order by a3;
-- 出现 using filesort
explain select * from test02 where a2='' order by a3;
-- 不出现
explain select * from test02 where a1='' order by a2;

-- 避免: where和order by按照复合索引的顺序使用,不要跨列或无序使用。
  • using temporary
explain select a1 from test02 where a1 in('1','2','3') group by a1;
explain select a1 from test02 where a1 in('1','2','3') group by a2;
-- 避免:查询哪些列,就用哪些列分组

8.4 show profile分析SQL

​ Mysql从5.0.37版本开始增加了对 show profiles 和 show profile 语句的支持。show profiles 能够在做SQL优化时帮助我们了解时间都耗费到哪里去了。

通过 have_profiling 参数,能够看到当前MySQL是否支持profile:

mysql> select @@have_profiling;
+------------------+
| @@have_profiling |
+------------------+
| YES              |
+------------------+
1 row in set, 1 warning (0.00 sec)

默认profiling是关闭的,可以通过set语句在Session级别开启profiling:

mysql> select @@profiling;
+-------------+
| @@profiling |
+-------------+
|           0 |
+-------------+
1 row in set, 1 warning (0.00 sec)

-- 开启profiling
set profiling = 1;

通过profile,我们能够更清楚地了解SQL执行的过程。

首先,我们可以执行一系列的操作,如下图所示:

show databases;
mysql> use sql_optimization;
mysql> show tables;
mysql> select * from t_user;

执行完上述命令之后,再执行show profiles 指令, 来查看SQL语句执行的耗时:

mysql> show profiles;
+----------+------------+----------------------+
| Query_ID | Duration   | Query                |
+----------+------------+----------------------+
|        1 | 0.00059300 | show databases       |
|        2 | 0.00014500 | SELECT DATABASE()    |
|        3 | 0.00041800 | show tables          |
|        4 | 0.00151500 | select * from t_user |
+----------+------------+----------------------+

通过show profile for query query_id 语句可以查看到该SQL执行过程中每个线程的状态和消耗的时间:

mysql> show profile for query 4;
+----------------------+----------+
| Status               | Duration |
+----------------------+----------+
| starting             | 0.000066 |
| checking permissions | 0.000011 |
| Opening tables       | 0.000022 |
| init                 | 0.000022 |
| System lock          | 0.000011 |
| optimizing           | 0.000005 |
| statistics           | 0.001128 |
| preparing            | 0.000034 |
| executing            | 0.000005 |
| Sending data         | 0.000123 |
| end                  | 0.000009 |
| query end            | 0.000009 |
| closing tables       | 0.000012 |
| freeing items        | 0.000035 |
| cleaning up          | 0.000023 |
+----------------------+----------+
TIP:
	Sending data 状态表示MySQL线程开始访问数据行并把结果返回给客户端,而不仅仅是返回给客户端,由于在Sending data状态下,MySQL线程往往需要做大量的磁盘读取操作,所以经常是整个查询中耗时最长的状态。

在获取到最消耗时间的线程状态后,MySQL支持进一步选择all、cpu、block io 、context switch、page faults等
明细类型类查看MySQL在使用什么资源上耗费了过高的时间。例如,选择查看CPU的耗费时间 :

mysql> show profile cpu for query 4;
+----------------------+----------+----------+------------+
| Status               | Duration | CPU_user | CPU_system |
+----------------------+----------+----------+------------+
| starting             | 0.000066 | 0.000057 |   0.000009 |
| checking permissions | 0.000011 | 0.000009 |   0.000003 |
| Opening tables       | 0.000022 | 0.000019 |   0.000001 |
| init                 | 0.000022 | 0.000021 |   0.000002 |
| System lock          | 0.000011 | 0.000009 |   0.000002 |
| optimizing           | 0.000005 | 0.000004 |   0.000001 |
| statistics           | 0.001128 | 0.000030 |   0.000222 |
| preparing            | 0.000034 | 0.000024 |   0.000009 |
| executing            | 0.000005 | 0.000002 |   0.000002 |
| Sending data         | 0.000123 | 0.000091 |   0.000017 |
| end                  | 0.000009 | 0.000004 |   0.000004 |
| query end            | 0.000009 | 0.000008 |   0.000002 |
| closing tables       | 0.000012 | 0.000011 |   0.000001 |
| freeing items        | 0.000035 | 0.000012 |   0.000022 |
| cleaning up          | 0.000023 | 0.000023 |   0.000016 |
+----------------------+----------+----------+------------+

-- status: sql语句执行的状态
-- duration: sql执行过程中每一个步骤的耗时
-- cpu_user: 当前用户占有的cpu
-- cpu_system: 系统占有的cpu

8.5 trace分析优化器执行计划

MySQL5.6提供了对SQL的跟踪trace, 通过trace文件能够进一步了解为什么优化器选择A计划, 而不是选择B计划。

打开trace , 设置格式为 JSON,并设置trace最大能够使用的内存大小,避免解析过程中因为默认内存过小而不能
够完整展示。

set optimizer_trace="enabled=on",end_markers_in_json=on; 
set optimizer_trace_max_mem_size=1000000;

执行SQL语句 :

select * from t_user where id <5;

最后, 检查information_schema.optimizer_trace就可以知道MySQL是如何执行SQL的 :

select * from information_schema.optimizer_trace\G;
*************************** 1. row ***************************
QUERY: select * from t_user where id <5
TRACE: {
  "steps": [
    {
      "join_preparation": {
        "select#": 1,
        "steps": [
          {
            "expanded_query": "/* select#1 */ select `t_user`.`id` AS `id`,`t_user`.`username` AS `username`,`t_user`.`password` AS `password`,`t_user`.`name` AS `name` from `t_user` where (`t_user`.`id` < 5)"
          }
        ] /* steps */
      } /* join_preparation */
    },
    {
      "join_optimization": {
        "select#": 1,
        "steps": [
          {
            "condition_processing": {
              "condition": "WHERE",
              "original_condition": "(`t_user`.`id` < 5)",
              "steps": [
                {
                  "transformation": "equality_propagation",
                  "resulting_condition": "(`t_user`.`id` < 5)"
                },
                {
                  "transformation": "constant_propagation",
                  "resulting_condition": "(`t_user`.`id` < 5)"
                },
                {
                  "transformation": "trivial_condition_removal",
                  "resulting_condition": "(`t_user`.`id` < 5)"
                }
              ] /* steps */
            } /* condition_processing */
          },
          {
            "substitute_generated_columns": {
            } /* substitute_generated_columns */
          },
          {
            "table_dependencies": [
              {
                "table": "`t_user`",
                "row_may_be_null": false,
                "map_bit": 0,
                "depends_on_map_bits": [
                ] /* depends_on_map_bits */
              }
            ] /* table_dependencies */
          },
          {
            "ref_optimizer_key_uses": [
            ] /* ref_optimizer_key_uses */
          },
          {
            "rows_estimation": [
              {
                "table": "`t_user`",
                "range_analysis": {
                  "table_scan": {
                    "rows": 6,
                    "cost": 4.3
                  } /* table_scan */,
                  "potential_range_indexes": [
                    {
                      "index": "PRIMARY",
                      "usable": true,
                      "key_parts": [
                        "id"
                      ] /* key_parts */
                    },
                    {
                      "index": "unique_user_username",
                      "usable": false,
                      "cause": "not_applicable"
                    }
                  ] /* potential_range_indexes */,
                  "setup_range_conditions": [
                  ] /* setup_range_conditions */,
                  "group_index_range": {
                    "chosen": false,
                    "cause": "not_group_by_or_distinct"
                  } /* group_index_range */
                } /* range_analysis */
              }
            ] /* rows_estimation */
          },
          {
            "considered_execution_plans": [
              {
                "plan_prefix": [
                ] /* plan_prefix */,
                "table": "`t_user`",
                "best_access_path": {
                  "considered_access_paths": [
                    {
                      "rows_to_scan": 6,
                      "access_type": "scan",
                      "resulting_rows": 6,
                      "cost": 2.2,
                      "chosen": true
                    }
                  ] /* considered_access_paths */
                } /* best_access_path */,
                "condition_filtering_pct": 100,
                "rows_for_plan": 6,
                "cost_for_plan": 2.2,
                "chosen": true
              }
            ] /* considered_execution_plans */
          },
          {
            "attaching_conditions_to_tables": {
              "original_condition": "(`t_user`.`id` < 5)",
              "attached_conditions_computation": [
              ] /* attached_conditions_computation */,
              "attached_conditions_summary": [
                {
                  "table": "`t_user`",
                  "attached": "(`t_user`.`id` < 5)"
                }
              ] /* attached_conditions_summary */
            } /* attaching_conditions_to_tables */
          },
          {
            "refine_plan": [
              {
                "table": "`t_user`"
              }
            ] /* refine_plan */
          }
        ] /* steps */
      } /* join_optimization */
    },
    {
      "join_execution": {
        "select#": 1,
        "steps": [
        ] /* steps */
      } /* join_execution */
    }
  ] /* steps */
}

9、索引的使用

​ 索引是数据库优化最常用也是最重要的手段之一, 通过索引通常可以帮助用户解决大多数的MySQL的性能优化问题。

9.1索引的使用

9.1.1 准备环境
create table `tb_seller` ( `sellerid` varchar (100), `name` varchar (100), `nickname` varchar (50), `password` varchar (60), `status` varchar (1), `address` varchar (100), `createtime` datetime, primary key(`sellerid`)
)engine=innodb default charset=utf8mb4;


insert into `tb_seller` (`sellerid`, `name`, `nickname`, `password`, `status`, `address`, `createtime`) values('alibaba','阿里巴巴','阿里小店','e10adc3949ba59abbe56e057f20f883e','1','北京市','2088-01-01 12:00:00'); insert into `tb_seller` (`sellerid`, `name`, `nickname`, `password`, `status`, `address`, `createtime`) values('baidu','百度科技有限公司','百度小店','e10adc3949ba59abbe56e057f20f883e','1','北京市','2088-01-01 12:00:00'); insert into `tb_seller` (`sellerid`, `name`, `nickname`, `password`, `status`, `address`, `createtime`) values('huawei','华为科技有限公司','华为小店','e10adc3949ba59abbe56e057f20f883e','0','北京市','2088-01-01 12:00:00');
insert into `tb_seller` (`sellerid`, `name`, `nickname`, `password`, `status`, `address`, `createtime`) values('itcast','锤子科技有限公司','锤子科技','e10adc3949ba59abbe56e057f20f883e','1','北京市','2088-01-01 12:00:00');
insert into `tb_seller` (`sellerid`, `name`, `nickname`, `password`, `status`, `address`, `createtime`) values('itheima','腾讯科技有限公司','腾讯科技','e10adc3949ba59abbe56e057f20f883e','0','北京市','2088-01-01 12:00:00'); insert into `tb_seller` (`sellerid`, `name`, `nickname`, `password`, `status`, `address`, `createtime`) values('luoji','罗技科技有限公司','罗技小店','e10adc3949ba59abbe56e057f20f883e','1','北京市','2088-01-01 12:00:00'); insert into `tb_seller` (`sellerid`, `name`, `nickname`, `password`, `status`, `address`, `createtime`) values('oppo','OPPO科技有限公司','OPPO官方旗舰店','e10adc3949ba59abbe56e057f20f883e','0','北京市','2088-01-01 12:00:00'); insert into `tb_seller` (`sellerid`, `name`, `nickname`, `password`, `status`, `address`, `createtime`) values('ourpalm','掌趣科技股份有限公司','掌趣小店','e10adc3949ba59abbe56e057f20f883e','1','北京市','2088-01-01 12:00:00'); insert into `tb_seller` (`sellerid`, `name`, `nickname`, `password`, `status`, `address`, `createtime`) values('qiandu','千度科技','千度小店','e10adc3949ba59abbe56e057f20f883e','2','北京市','2088-01-01 12:00:00'); insert into `tb_seller` (`sellerid`, `name`, `nickname`, `password`, `status`, `address`, `createtime`) values('sina','新浪科技有限公司','新浪官方旗舰店','e10adc3949ba59abbe56e057f20f883e','1','北京市','2088-01-01 12:00:00'); insert into `tb_seller` (`sellerid`, `name`, `nickname`, `password`, `status`, `address`, `createtime`) values('xiaomi','小米科技','小米官方旗舰店','e10adc3949ba59abbe56e057f20f883e','1','西安市','2088-01-01 12:00:00'); insert into `tb_seller` (`sellerid`, `name`, `nickname`, `password`, `status`, `address`, `createtime`) values('yijia','宜家家居','宜家家居旗舰店','e10adc3949ba59abbe56e057f20f883e','1','北京市','2088-01-01 12:00:00');

create index idx_seller_name_sta_addr on tb_seller(name,status,address);
9.1.2 避免索引失效

1、全值匹配,对索引中所有列都指定具体值。

  • 该情况下,索引生效,执行效率高
explain select * from tb_seller where name='小米科技' and status='1' and address='北京市'\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: tb_seller
   partitions: NULL
         type: ref
possible_keys: idx_seller_name_sta_addr
          key: idx_seller_name_sta_addr
      key_len: 813
          ref: const,const,const
         rows: 1
     filtered: 100.00
        Extra: NULL

2、最左前缀(与where后接的条件顺序无关,包含最左索引字段,则满足最左前缀原则)

  • 如果索引了多列,要遵守最左前缀法则。指的是查询从索引的最左前列开始,并且不跳过索引中的列。
mysql> explain select * from tb_seller where name='小米科技'\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: tb_seller
   partitions: NULL
         type: ref
possible_keys: idx_seller_name_sta_addr
          key: idx_seller_name_sta_addr
      key_len: 403
          ref: const
         rows: 1
     filtered: 100.00
        Extra: NULL
        
mysql> explain select * from tb_seller where name='小米科技' and status='1'\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: tb_seller
   partitions: NULL
         type: ref
possible_keys: idx_seller_name_sta_addr
          key: idx_seller_name_sta_addr
      key_len: 410
          ref: const,const
         rows: 1
     filtered: 100.00
        Extra: NULL
        
mysql> explain select * from tb_seller where name='小米科技' and address='西安市'\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: tb_seller
   partitions: NULL
         type: ref
possible_keys: idx_seller_name_sta_addr
          key: idx_seller_name_sta_addr
      key_len: 403
          ref: const
         rows: 1
     filtered: 10.00
        Extra: Using index condition
        
 -- 最左前缀与where后接的查询条件无关,条件包含了最左前缀即可      
mysql> explain select * from tb_seller where status='1' and name='小米科技'\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: tb_seller
   partitions: NULL
         type: ref
possible_keys: idx_seller_name_sta_addr
          key: idx_seller_name_sta_addr
      key_len: 410
          ref: const,const
         rows: 1
     filtered: 100.00
        Extra: NULL
        
-- 需要回表查询
mysql> explain select password from tb_seller where password ='e10adc3949ba59abbe56e057f20f883e'\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: tb_seller
   partitions: NULL
         type: ALL
possible_keys: NULL
          key: NULL
      key_len: NULL
          ref: NULL
         rows: 12
     filtered: 10.00
        Extra: Using where

3、范围查询右边的列,不能使用索引

-- 根据key_len判断states右边的
mysql> explain select * from tb_seller where name='小米科技' and status>'1'  and address='西安市'\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: tb_seller
   partitions: NULL
         type: range
possible_keys: idx_seller_name_sta_addr
          key: idx_seller_name_sta_addr
      key_len: 410
          ref: NULL
         rows: 1
     filtered: 10.00
        Extra: Using index condition
1 row in set, 1 warning (0.00 sec)


mysql> explain select * from tb_seller where name='小米科技' and status='1'  and address='西安市'\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: tb_seller
   partitions: NULL
         type: ref
possible_keys: idx_seller_name_sta_addr
          key: idx_seller_name_sta_addr
      key_len: 813
          ref: const,const,const
         rows: 1
     filtered: 100.00
        Extra: NULL
1 row in set, 1 warning (0.00 sec)

4、不要再索引列上进行运算操作,否则索引失效

mysql> explain select * from tb_seller where substring(name,3,2)='科技'\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: tb_seller
   partitions: NULL
         type: ALL
possible_keys: NULL
          key: NULL
      key_len: NULL
          ref: NULL
         rows: 12
     filtered: 100.00
        Extra: Using where

5、字符串不加单引号,造成索引失效

mysql> explain select * from tb_seller where name='小米科技' and status=1\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: tb_seller
   partitions: NULL
         type: ref
possible_keys: idx_seller_name_sta_addr
          key: idx_seller_name_sta_addr
      key_len: 403
          ref: const
         rows: 1
     filtered: 10.00
        Extra: Using index condition
        
mysql> explain select * from tb_seller where name='小米科技' and status='1'\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: tb_seller
   partitions: NULL
         type: ref
possible_keys: idx_seller_name_sta_addr
          key: idx_seller_name_sta_addr
      key_len: 410
          ref: const,const
         rows: 1
     filtered: 100.00
        Extra: NULL

由于在查询时,没有对字符串加单引号,MySQL的查询优化器,会自动的进行类型转换,造成索引失效。

6、尽量使用覆盖索引,避免select *

mysql> explain select * from tb_seller where name='小米科技' and status='1'\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: tb_seller
   partitions: NULL
         type: ref
possible_keys: idx_seller_name_sta_addr
          key: idx_seller_name_sta_addr
      key_len: 410
          ref: const,const
         rows: 1
     filtered: 100.00
        Extra: NULL

mysql> explain select name,status,address from tb_seller where name='小米科技' and status='1'\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: tb_seller
   partitions: NULL
         type: ref
possible_keys: idx_seller_name_sta_addr
          key: idx_seller_name_sta_addr
      key_len: 410
          ref: const,const
         rows: 1
     filtered: 100.00
        Extra: Using index       

如果查询列,超出索引列,也会降低性能

mysql> explain select name,status,address,password from tb_seller where name='小米科技' and status='1'\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: tb_seller
   partitions: NULL
         type: ref
possible_keys: idx_seller_name_sta_addr
          key: idx_seller_name_sta_addr
      key_len: 410
          ref: const,const
         rows: 1
     filtered: 100.00
        Extra: NULL

7、用or分割开的条件,如果or前面的列有索引,而后面的列没有索引,那么涉及的索引都不会被用到。

示例:name 字段是索引列,而nickname不是索引列,中间or进行连接是不走索引的:

mysql> explain select * from tb_seller where name='小米科技' or nickname='小米官方旗舰店'\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: tb_seller
   partitions: NULL
         type: ALL
possible_keys: idx_seller_name_sta_addr
          key: NULL
      key_len: NULL
          ref: NULL
         rows: 12
     filtered: 19.00
        Extra: Using where

8、以%开头的like模糊查询,索引失效

如果仅尾部模糊匹配,索引不失效,如果是头部模糊匹配,索引失效

mysql> explain select * from tb_seller where name like '%小米%'\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: tb_seller
   partitions: NULL
         type: ALL
possible_keys: NULL
          key: NULL
      key_len: NULL
          ref: NULL
         rows: 12
     filtered: 11.11
        Extra: Using where
        
explain select * from tb_seller where name like '%科技'\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: tb_seller
   partitions: NULL
         type: ALL
possible_keys: NULL
          key: NULL
      key_len: NULL
          ref: NULL
         rows: 12
     filtered: 11.11
        Extra: Using where
        
explain select * from tb_seller where name like '小米%'\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: tb_seller
   partitions: NULL
         type: range
possible_keys: idx_seller_name_sta_addr
          key: idx_seller_name_sta_addr
      key_len: 403
          ref: NULL
         rows: 1
     filtered: 100.00
        Extra: Using index condition      

解决办法:

通过覆盖索引解决

mysql> explain select name,status,address from tb_seller where name like '%科技%'\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: tb_seller
   partitions: NULL
         type: index
possible_keys: NULL
          key: idx_seller_name_sta_addr
      key_len: 813
          ref: NULL
         rows: 12
     filtered: 11.11
        Extra: Using where; Using index

9、如果mysql评估使用索引比全表更慢,则不使用索引

mysql> create index idx_address on tb_seller(address);

mysql> explain select * from tb_seller where address='西安市'\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: tb_seller
   partitions: NULL
         type: ref
possible_keys: idx_address
          key: idx_address
      key_len: 403
          ref: const
         rows: 1
     filtered: 100.00
        Extra: NULL
        
mysql> explain select * from tb_seller where address='北京市'\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: tb_seller
   partitions: NULL
         type: ALL
possible_keys: idx_address
          key: NULL
      key_len: NULL
          ref: NULL
         rows: 12
     filtered: 91.67
        Extra: Using where

根据数据库表数据有关,被索引字段的值大部门都是一样,则数据库优化引擎不会使用索引,而进行全表扫描

10、is NULL 、is NOT NULL 有时索引失效(根据数据库的数据量决定的)

和数据库表中的数据有关

  • 如果大部分数据为null

    • is null 索引失效,数据库会自动选择全表扫描
    • is not null 使用索引
  • 如果大部分数据不为null

    • is not null 数据库会自动选择全表扫描,索引失效
    • is null 使用索引

11、in走索引,not in 索引失效

mysql> explain select * from tb_seller where sellerid in('xiaomi','baidu','alibaba')\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: tb_seller
   partitions: NULL
         type: range
possible_keys: PRIMARY
          key: PRIMARY
      key_len: 402
          ref: NULL
         rows: 3
     filtered: 100.00
        Extra: Using where
        
 
mysql> explain select * from tb_seller where sellerid not in('xiaomi','baidu','alibaba')\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: tb_seller
   partitions: NULL
         type: ALL
possible_keys: PRIMARY
          key: NULL
      key_len: NULL
          ref: NULL
         rows: 12
     filtered: 91.67
        Extra: Using where       

12、单列索引和复合索引

尽量使用复合索引,而少使用单列索引

创建复合索引

create index idx_name_status_address on tb_seller(name,status,address);

就相当于创建了三个索引:
		name
		name+status
		name+status+address

创建单列索引

create index idx_seller_name on tb_seller(name); 
create index idx_seller_status on tb_seller(status); 
create index idx_seller_address on tb_seller(address);

数据库会选择一个最优的索引(辨识度最高的索引)来使用,并不会使用全部索引。

-- 只用到了idx_seller_name索引('小米科技'在表里是唯一的,辨识度高)
mysql> explain select * from tb_seller where name='小米科技' and status='1' and address='西安市'\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: tb_seller
   partitions: NULL
         type: ref
possible_keys: idx_seller_name,idx_seller_status,idx_seller_address
          key: idx_seller_name
      key_len: 403
          ref: const
         rows: 1
     filtered: 8.33
        Extra: Using where

-- 只用到了idx_seller_name索引('西安市'在表中只有一条数据,辨识度高)
mysql> explain select * from tb_seller where status='0' and address='西安市'\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: tb_seller
   partitions: NULL
         type: ref
possible_keys: idx_seller_status,idx_seller_address
          key: idx_seller_address
      key_len: 403
          ref: const
         rows: 1
     filtered: 66.67
        Extra: Using where
        
-- 只用到了idx_seller_status索引(status辨识度高)
mysql> explain select * from tb_seller where status='0' and address='北京市'\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: tb_seller
   partitions: NULL
         type: ref
possible_keys: idx_seller_status,idx_seller_address
          key: idx_seller_status
      key_len: 7
          ref: const
         rows: 3
     filtered: 91.67
        Extra: Using where

9.2 查看索引的使用情况

-- 当前会话
show status like 'Handler_read%';
-- 全局
show global status like 'Handler_read%';
Handler_read_first:索引中第一条被读的次数。如果较高,表示服务器正执行大量全索引扫描(这个值越低 越好)。
Handler_read_key:如果索引正在工作,这个值代表一个行被索引值读的次数,如果值越低,表示索引得到的 性能改善不高,因为索引不经常使用(这个值越高越好)。
Handler_read_next :按照键顺序读下一行的请求数。如果你用范围约束或如果执行索引扫描来查询索引列, 该值增加。
Handler_read_prev:按照键顺序读前一行的请求数。该读方法主要用于优化ORDER BY ... DESC。
Handler_read_rnd :根据固定位置读一行的请求数。如果你正执行大量查询并需要对结果进行排序该值较高。 你可能使用了大量需要MySQL扫描整个表的查询或你的连接没有正确使用键。这个值较高,意味着运行效率低,应 该建立索引来补救。
Handler_read_rnd_next:在数据文件中读下一行的请求数。如果你正进行大量的表扫描,该值较高。通常说明你的表索引不正确或写入的查询没有利用索引。

10 SQL优化

10.1 大批量插入数据

环境准备:

CREATE TABLE `tb_user1` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`username` varchar(45) NOT NULL,
`password` varchar(96) NOT NULL,
`name` varchar(45) NOT NULL,
`birthday` datetime DEFAULT NULL,
`sex` char(1) DEFAULT NULL,
`email` varchar(45) DEFAULT NULL,
`phone` varchar(45) DEFAULT NULL,
`qq` varchar(32) DEFAULT NULL,
`status` varchar(32) NOT NULL COMMENT '用户状态', `create_time` datetime NOT NULL,
`update_time` datetime DEFAULT NULL,
PRIMARY KEY (`id`),
UNIQUE KEY `unique_user_username` (`username`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 ;

CREATE TABLE `tb_user2` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`username` varchar(45) NOT NULL,
`password` varchar(96) NOT NULL,
`name` varchar(45) NOT NULL,
`birthday` datetime DEFAULT NULL,
`sex` char(1) DEFAULT NULL,
`email` varchar(45) DEFAULT NULL,
`phone` varchar(45) DEFAULT NULL,
`qq` varchar(32) DEFAULT NULL,
`status` varchar(32) NOT NULL COMMENT '用户状态', `create_time` datetime NOT NULL,
`update_time` datetime DEFAULT NULL,
PRIMARY KEY (`id`),
UNIQUE KEY `unique_user_username` (`username`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 ;

当使用load命令导入数据的时候,适当的设置可以提高查询效率

对于innoDB类型的表,有以下几种方式可以提高导入的效率:

1)主键顺序插入

因为InnoDB类型的表是按照主键的顺序保存的,所以将导入的数据按照主键的顺序排列,可以有效的提高导入数据的效率。如果InnoDB表没有主键,那么系统会自动默认创建一个内部列作为主键,所以如果可以给表创建一个
主键,将可以利用这点,来提高导入数据的效率。

脚本文件介绍 :
sql1.log ----> 主键有序
sql2.log ----> 主键无序

插入ID顺序排列的数据:

-- 主键有序插入数据
mysql> LOAD DATA LOCAL INFILE '/Users/mac/Documents/sql1.log' INTO TABLE tb_user1 FIELDS TERMINATED BY ',' LINES TERMINATED BY '\n';
Query OK, 1000000 rows affected, 65535 warnings (17.03 sec)

-- 主键无序插入数据
mysql> LOAD DATA LOCAL INFILE '/Users/mac/Documents/sql2.log' INTO TABLE tb_user2 FIELDS TERMINATED BY ',' LINES TERMINATED BY '\n';
Query OK, 1000000 rows affected, 65535 warnings (43.92 sec)

2)关闭唯一性校验

在导入数据前执行 SET UNIQUE_CHECKS=0,关闭唯一性校验,在导入结束后执行SET UNIQUE_CHECKS=1,恢复唯一性校验。(关闭唯一性校验,可以提高导入的效率

mysql> SET UNIQUE_CHECKS=0;

mysql> LOAD DATA LOCAL INFILE '/Users/mac/Documents/sql1.log' INTO TABLE tb_user1 FIELDS TERMINATED BY ',' LINES TERMINATED BY '\n';
Query OK, 1000000 rows affected, 65535 warnings (17.03 sec)

mysql> SET UNIQUE_CHECKS=1;

3)手动提交事物

如果应用使用自动提交的方式,建议在导入前执行SET AUTOCOMMIT=0,关闭自动提交,导入结束后在执行 SET AUTOCOMMIT=1,打开自动提交。(手动提交事务,可以提高导入效率

10.2 优化insert语句

当进行数据的insert操作的时候,可以考虑采用以下几种优化方案。

  • 往一张表插入多行数据

    示例,原始方式

    insert into tb_test values(1,'Tom'); 
    insert into tb_test values(2,'Cat'); 
    insert into tb_test values(3,'Jerry');
    

    优化方案:

    insert into tb_test values(1,'Tom'),(2,'Cat'),(3,'Jerry');
    
  • 在事务中进行数据插入

    start transaction;
    insert into tb_test values(1,'Tom'); 
    insert into tb_test values(2,'Cat'); 
    insert into tb_test values(3,'Jerry'); 
    commit;
    
  • 数据有序插入

    insert into tb_test values(4,'Tim');
    insert into tb_test values(1,'Tom'); 
    insert into tb_test values(3,'Jerry'); 
    insert into tb_test values(5,'Rose'); 
    insert into tb_test values(2,'Cat');
    

    优化方案:

    insert into tb_test values(1,'Tom'); 
    insert into tb_test values(2,'Cat'); 
    insert into tb_test values(3,'Jerry'); 
    insert into tb_test values(4,'Tim');
    insert into tb_test values(5,'Rose'); 
    

10.3 优化order by语句

10.3.1 环境准备
CREATE TABLE `emp` (
`id` int(11) NOT NULL AUTO_INCREMENT, `name` varchar(100) NOT NULL,
`age` int(3) NOT NULL,
`salary` int(11) DEFAULT NULL, 
  PRIMARY KEY (`id`)
) ENGINE=InnoDB  DEFAULT CHARSET=utf8mb4;
insert into `emp` (`id`, `name`, `age`, `salary`) values('1','Tom','25','2300');
insert into `emp` (`id`, `name`, `age`, `salary`) values('2','Jerry','30','3500');
insert into `emp` (`id`, `name`, `age`, `salary`) values('3','Luci','25','2800');
insert into `emp` (`id`, `name`, `age`, `salary`) values('4','Jay','36','3500');
insert into `emp` (`id`, `name`, `age`, `salary`) values('5','Tom2','21','2200');
insert into `emp` (`id`, `name`, `age`, `salary`) values('6','Jerry2','31','3300');
insert into `emp` (`id`, `name`, `age`, `salary`) values('7','Luci2','26','2700');
insert into `emp` (`id`, `name`, `age`, `salary`) values('8','Jay2','33','3500');
insert into `emp` (`id`, `name`, `age`, `salary`) values('9','Tom3','23','2400');
insert into `emp` (`id`, `name`, `age`, `salary`)
values('10','Jerry3','32','3100');
insert into `emp` (`id`, `name`, `age`, `salary`) values('11','Luci3','26','2900');
insert into `emp` (`id`, `name`, `age`, `salary`) values('12','Jay3','37','4500');
create index idx_emp_age_salary on emp(age,salary);
10.3.2 俩种排序方式

1)通过对返回数据进行排序,也就是通常说的filesort排序,所有不是通过索引直接返回排序结果的排序都叫 FileSort排序

mysql> explain select * from emp order by age asc\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: emp
   partitions: NULL
         type: ALL
possible_keys: NULL
          key: NULL
      key_len: NULL
          ref: NULL
         rows: 12
     filtered: 100.00
        Extra: Using filesort

mysql> explain select * from emp order by age desc\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: emp
   partitions: NULL
         type: ALL
possible_keys: NULL
          key: NULL
      key_len: NULL
          ref: NULL
         rows: 12
     filtered: 100.00
        Extra: Using filesort

2)通过有序索引顺序扫描直接返回有序数据,这种情况即为using index,不需要额外排序,操作效率高。

mysql> explain select id,age from emp order by age desc\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: emp
   partitions: NULL
         type: index
possible_keys: NULL
          key: idx_emp_age_salary
      key_len: 9
          ref: NULL
         rows: 12
     filtered: 100.00
        Extra: Using index
        
mysql> explain select id,age,name from emp order by age desc\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: emp
   partitions: NULL
         type: ALL
possible_keys: NULL
          key: NULL
      key_len: NULL
          ref: NULL
         rows: 12
     filtered: 100.00
        Extra: Using filesort

多字段排序

mysql> explain select id,age from emp order by age desc,salary desc\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: emp
   partitions: NULL
         type: index
possible_keys: NULL
          key: idx_emp_age_salary
      key_len: 9
          ref: NULL
         rows: 12
     filtered: 100.00
        Extra: Using index

mysql> explain select id,age from emp order by age asc,salary desc\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: emp
   partitions: NULL
         type: index
possible_keys: NULL
          key: idx_emp_age_salary
      key_len: 9
          ref: NULL
         rows: 12
     filtered: 100.00
        Extra: Using index; Using filesort
1 row in set, 1 warning (0.00 sec)

mysql> explain select id,age from emp order by salary asc,age asc\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: emp
   partitions: NULL
         type: index
possible_keys: NULL
          key: idx_emp_age_salary
      key_len: 9
          ref: NULL
         rows: 12
     filtered: 100.00
        Extra: Using index; Using filesort

排序优化:

  • 尽量减少额外的排序,通过索引直接返回有序数据
  • where条件和order by使用相同的索引
  • order by 顺序与索引顺序相同
  • order by的字段都是升序或者都是降序
10.3.3 Filesort的优化

通过创建合适的索引,能够减少 Filesort 的出现,但是在某些情况下,条件限制不能让Filesort消失,那就需要加

快 Filesort的排序操作。对于Filesort , MySQL 有两种排序算法:

  1. 两次扫描算法 :MySQL4.1 之前,使用该方式排序。首先根据条件取出排序字段和行指针信息,然后在排序区 sort buffer 中排序,如果sort buffer不够,则在临时表 temporary table 中存储排序结果。完成排序之后,再根据 行指针回表读取记录,该操作可能会导致大量随机I/O操作。

2)一次扫描算法:一次性取出满足条件的所有字段,然后在排序区 sort buffer 中排序后直接输出结果集。排序时内存开销较大,但是排序效率比两次扫描算法要高。

MySQL 通过比较系统变量 max_length_for_sort_data 的大小和Query语句取出的字段总大小, 来判定是否那种排
序算法,如果max_length_for_sort_data 更大,那么使用第二种优化之后的算法;否则使用
第一种。
可以适当提高 sort_buffer_size 和 max_length_for_sort_data 系统变量,来增大排序区的大小,提高排序的效率。

show variables like 'max_length_for_sort_data';
show variables like 'sort_buffer_size';

10.4 优化 group by语句

​ 由于GROUP BY 实际上也同样会进行排序操作,而且与ORDER BY 相比,GROUP BY 主要只是多了排序之后的分组操作。当然,如果在分组的时候还使用了其他的一些聚合函数,那么还需要一些聚合函数的计算。所以,在GROUP BY 的实现过程中,与 ORDER BY 一样也可以利用到索引。

如果查询包含 group by 但是用户想要避免排序结果的消耗, 则可以执行order by null 禁止排序。如下 :

mysql> drop index idx_emp_age_salary on emp;
mysql> explain select age,count(*) from emp group by age\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: emp
   partitions: NULL
         type: ALL
possible_keys: NULL
          key: NULL
      key_len: NULL
          ref: NULL
         rows: 12
     filtered: 100.00
        Extra: Using temporary; Using filesort

优化案例

mysql> explain select age,count(*) from emp group by age order by null\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: emp
   partitions: NULL
         type: ALL
possible_keys: NULL
          key: NULL
      key_len: NULL
          ref: NULL
         rows: 12
     filtered: 100.00
        Extra: Using temporary

从上面的例子可以看出,第一个SQL语句需要进行"filesort",而第二个SQL由于order by null 不需要进行"filesort", 而上文提过Filesort往往非常耗费时间。

继续优化(创建索引)

create index idx_emp_age_salary on emp(age,salary);
mysql> explain select age,count(*) from emp group by age order by null\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: emp
   partitions: NULL
         type: index
possible_keys: idx_emp_age_salary
          key: idx_emp_age_salary
      key_len: 9
          ref: NULL
         rows: 12
     filtered: 100.00
        Extra: Using index

10.5 优化嵌套查询

​ Mysql4.1版本之后,开始支持SQL的子查询。这个技术可以使用SELECT语句来创建一个单列的查询结果,然后把这个结果作为过滤条件用在另一个查询中。使用子查询可以一次性的完成很多逻辑上需要多个步骤才能完成的SQL操作,同时也可以避免事务或者表锁死,并且写起来也很容易。但是,有些情况下,子查询是可以被更高效的连接(JOIN)替代。

示例:查找角色的所有用户信息:

在这里插入图片描述

优化后:

在这里插入图片描述

连接查询(join)查询效率更高的原因是:mysql不需要在内存中创建临时表来完成这个逻辑上需要俩步的操作

10.6 优化OR条件

​ 对于包含OR的查询子句,如果要利用索引,则OR之间的每个条件列都必须用到索引 , 而且不能使用到复合索
引; 如果没有索引,则应该考虑增加索引。

-- type为index_merge
mysql> explain select * from emp where id = 1 or age = 30\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: emp
   partitions: NULL
         type: index_merge
possible_keys: PRIMARY,idx_emp_age_salary
          key: idx_emp_age_salary,PRIMARY
      key_len: 4,4
          ref: NULL
         rows: 2
     filtered: 100.00
        Extra: Using sort_union(idx_emp_age_salary,PRIMARY); Using where
1 row in set, 1 warning (0.00 sec)

ERROR: 
No query specified

-- type为range
mysql> explain select * from emp where id = 1 or id = 10\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: emp
   partitions: NULL
         type: range
possible_keys: PRIMARY
          key: PRIMARY
      key_len: 4
          ref: NULL
         rows: 2
     filtered: 100.00
        Extra: Using where

建议使用 union 替换 or :

在这里插入图片描述

我们来比较下重要指标,发现主要差别是 type 和 ref 这两项
type 显示的是访问类型,是较为重要的一个指标,结果值从好到坏依次是:

system > const > eq_ref > ref > fulltext > ref_or_null > index_merge > unique_subquery > index_subquery > range > index > ALL

UNION 语句的 type 值为 ref,OR 语句的 type 值为 range,可以看到这是一个很明显的差距
UNION 语句的 ref 值为 const,OR 语句的 type 值为 null,const 表示是常量值引用,非常快
这两项的差距就说明了 UNION 要优于 OR 。

10.7 优化分页查询

​ 一般分页查询时,通过创建覆盖索引能够比较好地提高性能。一个常见又非常头疼的问题就是 limit 900000,10 , 此时需要MySQL排序前900010 记录,仅仅返回900000 - 900010 的记录,其他记录丢弃,查询排序的代价非 常大 。

mysql> explain select * from tb_user1 limit 500000,10\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: tb_user1
   partitions: NULL
         type: ALL
possible_keys: NULL
          key: NULL
      key_len: NULL
          ref: NULL
         rows: 991881
     filtered: 100.00
        Extra: NULL
10.7.1 优化思路一

在索引上完成排序分页操作,最后根据主键关联回原表查询所需要的其他列内容。

在这里插入图片描述

10.7.2 优化思路二

该方案适用于主键自增的表,可以把Limit 查询转换成某个位置的查询 。

  • 若ID出现断层,此方案无效

在这里插入图片描述

10.8 使用SQL提示

​ SQL提示,是优化数据库的一个重要手段,简单来说,就是在SQL语句中加入一些人为的提示来达到优化操作的目

的。

10.8.1 USE INDEX

在查询语句表名的后面,添加use index 来提供希望mysql去参考的索引列表,就可以让mysql不在考虑其他的索引。

mysql> create index tb_seller_name_sta_addr on tb_seller(name,status,address);
-- mysql引擎默认选择的索引:idx_seller_name
mysql> mysql> explain select * from tb_seller where name='小米科技'\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: tb_seller
   partitions: NULL
         type: ref
possible_keys: idx_seller_name,tb_seller_name_sta_addr
          key: idx_seller_name
      key_len: 403
          ref: const
         rows: 1
     filtered: 100.00
        Extra: NULL

-- 希望mysql去参考的索引:tb_seller_name_sta_addr
mysql> explain select * from tb_seller use index(tb_seller_name_sta_addr) where name='小米科技'\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: tb_seller
   partitions: NULL
         type: ref
possible_keys: tb_seller_name_sta_addr
          key: tb_seller_name_sta_addr
      key_len: 403
          ref: const
         rows: 1
     filtered: 100.00
        Extra: NULL
10.8.2 IGNORE INDEX

​ 如果用户只是单纯的想让MySQL忽略一个或者多个索引,则可以使用 ignore index

-- 忽略tb_seller_name_sta_addr索引
mysql> explain select * from tb_seller ignore index(tb_seller_name_sta_addr) where name='小米科技'\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: tb_seller
   partitions: NULL
         type: ref
possible_keys: idx_seller_name
          key: idx_seller_name
      key_len: 403
          ref: const
         rows: 1
     filtered: 100.00
        Extra: NULL
-- 忽略idx_seller_name索引
mysql> explain select * from tb_seller ignore index(idx_seller_name) where name='小米科技'\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: tb_seller
   partitions: NULL
         type: ref
possible_keys: tb_seller_name_sta_addr
          key: tb_seller_name_sta_addr
      key_len: 403
          ref: const
         rows: 1
     filtered: 100.00
        Extra: NULL
10.8.3 FORCE INDEX

​ 为强制MySQL使用一个特定的索引,可在查询中使用 force index。

​ 上文提到当where后面的检索字段的在表中的数据大部分与条件一样时,则索引不生效,转为全表扫描

  • 即:在数据库中address字段的值,大部分为’北京市’时,则数据库引擎会选择全表扫描
-- 数据库引擎选择全表扫描,索引不生效(use index提供的参考,数据库不采纳)
mysql> explain select * from tb_seller use index(idx_seller_address) where address='北京市'\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: tb_seller
   partitions: NULL
         type: ALL
possible_keys: NULL
          key: NULL
      key_len: NULL
          ref: NULL
         rows: 12
     filtered: 10.00
        Extra: Using where

mysql> explain select * from tb_seller force index(idx_seller_address) where address='北京市'\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: tb_seller
   partitions: NULL
         type: ref
possible_keys: idx_seller_address
          key: idx_seller_address
      key_len: 403
          ref: const
         rows: 11
     filtered: 100.00
        Extra: NULL

MySQL的其他内容详见以下链接:

MySQL之MySQL安装、索引使用及视图

MySQL之触发器的使用

MySQL之存储过程及函数的使用

MySQL之存储引擎及SQL优化

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值