超详细图解!【MySQL进阶篇,应届生java开发面试

​ 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;`

SQL

`start transaction;

insert into goods_innodb(id,name)values(null,‘Meta20’);

commit;

1556075130115

测试,发现在InnoDB中是存在事务的 ;

外键约束

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

​ 下面两张表中 , country_innodb是父表 , country_id为主键索引,city_innodb表是子表,country_id字段为外键,对应于country_innodb表的主键country_id 。

SQL

`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);`

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

RESTRICT和NO ACTION相同, 是指限制在子表有关联记录的情况下, 父表不能更新;

CASCADE表示父表在更新或者删除时,更新或者删除子表对应的记录;

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

针对上面创建的两个表, 子表的外键指定是ON DELETE RESTRICT ON UPDATE CASCADE 方式的, 那么在主表删除记录的时候, 如果子表有对应记录, 则不允许删除, 主表在更新记录的时候, 如果子表有对应记录, 则子表对应更新 。

表中数据如下图所示 :

1556087540767

外键信息可以使用如下两种方式查看 :

SQL

show create table city_innodb ;

1556087611295

删除country_id为1 的country数据:

SQL

delete from country_innodb where country_id = 1;

image.png

更新主表country表的字段 country_id :

SQL

`update country_innodb set country_id = 100 where country_id = 1;

更新后, 子表的数据信息为 :

1556087793738

存储方式

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

①. 使用共享表空间存储, 这种方式创建的表的表结构保存在.frm文件中, 数据和索引保存在 innodb_data_home_dir 和 innodb_data_file_path定义的表空间中,可以是多个文件。

②. 使用多表空间存储, 这种方式创建的表的表结构仍然存在 .frm 文件中,但是每个表的数据和索引单独保存在 .ibd 中。

1556075336630

2.2.2 MyISAM

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

不支持事务

SQL

`create table goods_myisam(

id int NOT NULL AUTO_INCREMENT,

name varchar(20) NOT NULL,

primary key(id)

)ENGINE=myisam DEFAULT CHARSET=utf8;

通过测试,我们发现,在MyISAM存储引擎中,是没有事务控制的 ;

文件存储方式

每个MyISAM在磁盘上存储成3个文件,其文件名都和表名相同,但拓展名分别是 :

.frm (存储表定义);

.MYD(MYData , 存储数据);

.MYI(MYIndex , 存储索引);

2.2.3 MEMORY

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

2.2.4 MERGE

​ MERGE存储引擎是一组MyISAM表的组合,这些MyISAM表必须结构完全相同,MERGE表本身并没有存储数据,对MERGE类型的表可以进行查询、更新、删除操作,这些操作实际上是对内部的MyISAM表进行的。

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

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

下面是一个创建和使用MERGE表的示例 :

1). 创建3个测试表 order_1990, order_1991, order_all , 其中order_all是前两个表的MERGE表 :

SQL

create table order_1990(

order_id int ,

order_money double(10,2),

order_address varchar(50),

primary key (order_id)

)engine = myisam default charset=utf8;

create table order_1991(

order_id int ,

order_money double(10,2),

order_address varchar(50),

primary key (order_id)

)engine = myisam default charset=utf8;

create table order_all(

order_id int ,

order_money double(10,2),

order_address varchar(50),

primary key (order_id)

2). 分别向两张表中插入记录

SQL

insert into order_1990 values(1,100.0,‘北京’);

insert into order_1990 values(2,100.0,‘上海’);

insert into order_1991 values(10,200.0,‘北京’);

insert into order_1991 values(11,200.0,‘上海’)

3). 查询3张表中的数据。

order_1990中的数据 :

image.png

order_1991中的数据 :

image.png

order_all中的数据 :

1551408216185

4). 往order_all中插入一条记录 ,由于在MERGE表定义时,INSERT_METHOD 选择的是LAST,那么插入的数据会想最后一张表中插入。

SQL

insert into order_all values(100,10000.0,‘西安’);

1551408519889

2.3 存储引擎的选择

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

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

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

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

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

3. 优化SQL步骤

在应用的的开发过程中,由于初期数据量小,开发人员写 SQL 语句时更重视功能上的实现,但是当应用系统正式上线后,随着生产数据量的急剧增长,很多 SQL 语句开始逐渐显露出性能问题,对生产的影响也越来越大,此时这些有问题的 SQL 语句就成为整个系统性能的瓶颈,因此我们必须要对它们进行优化,本章将详细介绍在 MySQL 中优化 SQL 语句的方法。

当面对一个有 SQL 性能问题的数据库时,我们应该从何处入手来进行系统的分析,使得能够尽快定位问题 SQL 并尽快解决问题。

3.1 查看SQL执行频率

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

下面的命令显示了当前 session 中所有统计参数的值:

SQL

`show status like ‘Com_______’;

image.png

SQL

`show status like ‘Innodb_rows_%’;

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

| 参数 | 含义 |

| — | — |

| Com_select | 执行 select 操作的次数,一次查询只累加 1。 |

| Com_insert | 执行 INSERT 操作的次数,对于批量插入的 INSERT 操作,只累加一次。 |

| Com_update | 执行 UPDATE 操作的次数。 |

| Com_delete | 执行 DELETE 操作的次数。 |

| Innodb_rows_read | select 查询返回的行数。 |

| Innodb_rows_inserted | 执行 INSERT 操作插入的行数。 |

| Innodb_rows_updated | 执行 UPDATE 操作更新的行数。 |

| Innodb_rows_deleted | 执行 DELETE 操作删除的行数。 |

| Connections | 试图连接 MySQL 服务器的次数。 |

| Uptime | 服务器工作时间。 |

| Slow_queries | 慢查询的次数。 |

Com_*** : 这些参数对于所有存储引擎的表操作都会进行累计。

Innodb_*** : 这几个参数只是针对InnoDB 存储引擎的,累加的算法也略有不同。

3.2 定位低效率执行SQL

可以通过以下两种方式定位执行效率较低的 SQL 语句。

  • 慢查询日志 : 通过慢查询日志定位那些执行效率较低的 SQL 语句,用–log-slow-queries[=file_name]选项启动时,mysqld 写一个包含所有执行时间超过 long_query_time 秒的 SQL 语句的日志文件。具体可以查看本书第 26 章中日志管理的相关部分。

  • show processlist : 慢查询日志在查询结束以后才纪录,所以在应用反映执行效率出现问题的时候查询慢查询日志并不能定位问题,可以使用show processlist命令查看当前MySQL在进行的线程,包括线程的状态、是否锁表等,可以实时地查看 SQL 的执行情况,同时对一些锁表操作进行优化。

1556098544349

YAML

`1) id列,用户登录mysql时,系统分配的"connection_id",可以使用函数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语句,是判断问题语句的一个重要依据`

3.3 explain分析执行计划

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

查询SQL语句的执行计划 :

SQL

explain select * from tb_item where id = 1;

SQL

explain select * from tb_item where title = ‘阿尔卡特 (OT-979) 冰川白 联通3G手机3’;

| 字段 | 含义 |

| — | — |

| id | select查询的序列号,是一组数字,表示的是查询中执行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 | 执行情况的说明和描述 |

3.3.1 环境准备

1556122799330

SQL

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 10 10TJ4TmCdK.X4wv/tCqHW14.w70U3CC33CeVncD3SLmyMXMknstqKRe’,‘超级管理员’);

insert into t_user (id, username, password, name) values(‘2’,‘admin’,‘$2a 10 10 10TJ4TmCdK.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 10 10pLtt2KDAFpwTWLjNsmTEi.oU1yOZyIn9XkziK/y/spH5rftCpUMZa’,‘学生1’);

insert into t_user (id, username, password, name) values(‘5’,‘stu2’,‘$2a 10 10 10nxPKkYSez7uz2YQYUnwhR.z57km3yqKn3Hr/p1FR6ZKgc18u.Tvqm’,‘学生2’);

insert into t_user (id, username, password, name) values(‘6’,‘t1’,‘$2a 10 10 10TJ4TmCdK.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’) ;`

3.3.2 explain 之 id

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

1) id 相同表示加载表的顺序是从上到下。

SQL

explain select * from t_role r, t_user u, user_role ur where r.id = ur.role_id and u.id = ur.user_id ;

1556102471304

2) id 不同id值越大,优先级越高,越先被执行。

SQL

EXPLAIN SELECT * FROM t_role WHERE id = (SELECT role_id FROM user_role WHERE user_id = (SELECT id FROM t_user WHERE username = 'stu1'))

1556103009534

3) id 有相同,也有不同,同时存在。id相同的可以认为是一组,从上往下顺序执行;在所有的组中,id的值越大,优先级越高,越先执行。

SQL

EXPLAIN SELECT * FROM t_role r , (SELECT * FROM user_role ur WHERE ur.user_id = '2') a WHERE r.id = a.role_id ;

1556103294182

3.3.3 explain 之 select_type

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

| select_type | 含义 |

| — | — |

| SIMPLE | 简单的select查询,查询中不包含子查询或者UNION |

| PRIMARY | 查询中若包含任何复杂的子查询,最外层查询标记为该标识 |

| SUBQUERY | 在SELECT 或 WHERE 列表中包含了子查询 |

| DERIVED | 在FROM 列表中包含的子查询,被标记为 DERIVED(衍生) MYSQL会递归执行这些子查询,把结果放在临时表中 |

| UNION | 若第二个SELECT出现在UNION之后,则标记为UNION ; 若UNION包含在FROM子句的子查询中,外层SELECT将被标记为 : DERIVED |

| UNION RESULT | 从UNION表获取结果的SELECT |

3.3.4 explain 之 table

展示这一行的数据是关于哪一张表的

3.3.5 explain 之 type

type 显示的是访问类型,是较为重要的一个指标,可取值为:

| type | 含义 |

| — | — |

| NULL | MySQL不访问任何表,索引,直接返回结果 |

| system | 表只有一行记录(等于系统表),这是const类型的特例,一般不会出现 |

| const | 表示通过索引一次就找到了,const 用于比较primary key 或者 unique 索引。因为只匹配一行数据,所以很快。如将主键置于where列表中,MySQL 就能将该查询转换为一个常亮。const于将 “主键” 或 “唯一” 索引的所有部分与常量值进行比较 |

| eq_ref | 类似ref,区别在于使用的是唯一索引,使用主键的关联查询,关联查询出的记录只有一条。常见于主键或唯一索引扫描 |

| ref | 非唯一性索引扫描,返回匹配某个单独值的所有行。本质上也是一种索引访问,返回所有匹配某个单独值的所有行(多个) |

| range | 只检索给定返回的行,使用一个索引来选择行。 where 之后出现 between , < , > , in 等操作。 |

| index | index 与 ALL的区别为 index 类型只是遍历了索引树, 通常比ALL 快, ALL 是遍历数据文件。 |

| all | 将遍历全表以找到匹配的行 |

结果值从最好到最坏以此是:

PERL

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

system > const > eq_ref > ref > range > index > ALL`

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

3.3.6 explain 之 key

PROPERTIES

`possible_keys : 显示可能应用在这张表的索引, 一个或多个。

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

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

3.3.7 explain 之 rows

扫描行的数量。

3.3.8 explain 之 extra

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

| extra | 含义 |

| — | — |

| using filesort | 说明mysql会对数据使用一个外部的索引排序,而不是按照表内的索引顺序进行读取, 称为 “文件排序”, 效率低。 |

| using temporary | 使用了临时表保存中间结果,MySQL在对查询结果排序时使用临时表。常见于 order by 和 group by; 效率低 |

| using index | 表示相应的select操作使用了覆盖索引, 避免访问表的数据行, 效率不错。 |

3.4 show profile分析SQL

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

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

1552488401999

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

1552488372405

SQL

`set profiling=1; //开启profiling 开关;

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

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

SQL

`show databases;

use db01;

show tables;

select * from tb_item where id < 5;

select count(*) from tb_item;`

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

1552489017940

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

1552489053763

TEX

`TIP :

Sending data 状态表示MySQL线程开始访问数据行并把结果返回给客户端,而不仅仅是返回个客户端。由于在Sending data状态下,MySQL线程往往需要做大量的磁盘读取操作,所以经常是整各查询中耗时最长的状态。

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

1552489671119

| 字段 | 含义 |

| — | — |

| Status | sql 语句执行的状态 |

| Duration | sql 执行过程中每一个步骤的耗时 |

| CPU_user | 当前用户占有的cpu |

| CPU_system | 系统占有的cpu |

3.5 trace分析优化器执行计划

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

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

SQL

`SET optimizer_trace=“enabled=on”,end_markers_in_json=on;

set optimizer_trace_max_mem_size=1000000;`

执行SQL语句 :

SQL

select * from tb_item where id < 4;

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

SQL

select * from information_schema.optimizer_trace\G;

JSON

`*************************** 1. row ***************************

QUERY: select * from tb_item where id < 4

TRACE: {

“steps”: [

{

“join_preparation”: {

“select#”: 1,

“steps”: [

{

“expanded_query”: “/* select#1 */ select tb_item.id AS id,tb_item.title AS title,tb_item.price AS price,tb_item.num AS num,tb_item.categoryid AS categoryid,tb_item.status AS status,tb_item.sellerid AS sellerid,tb_item.createtime AS createtime,tb_item.updatetime AS updatetime from tb_item where (tb_item.id < 4)”

}

] /* steps */

} /* join_preparation */

},

{

“join_optimization”: {

“select#”: 1,

“steps”: [

{

“condition_processing”: {

“condition”: “WHERE”,

“original_condition”: “(tb_item.id < 4)”,

“steps”: [

{

“transformation”: “equality_propagation”,

“resulting_condition”: “(tb_item.id < 4)”

},

{

“transformation”: “constant_propagation”,

“resulting_condition”: “(tb_item.id < 4)”

},

{

“transformation”: “trivial_condition_removal”,

“resulting_condition”: “(tb_item.id < 4)”

}

] /* steps */

} /* condition_processing */

},

{

“table_dependencies”: [

{

“table”: “tb_item”,

“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”: “tb_item”,

“range_analysis”: {

“table_scan”: {

“rows”: 9816098,

“cost”: 2.04e6

} /* table_scan */,

“potential_range_indices”: [

{

“index”: “PRIMARY”,

“usable”: true,

“key_parts”: [

“id”

] /* key_parts */

}

] /* potential_range_indices */,

“setup_range_conditions”: [

] /* setup_range_conditions */,

“group_index_range”: {

“chosen”: false,

“cause”: “not_group_by_or_distinct”

} /* group_index_range */,

“analyzing_range_alternatives”: {

“range_scan_alternatives”: [

{

“index”: “PRIMARY”,

“ranges”: [

“id < 4”

] /* ranges */,

“index_dives_for_eq_ranges”: true,

“rowid_ordered”: true,

“using_mrr”: false,

“index_only”: false,

“rows”: 3,

“cost”: 1.6154,

“chosen”: true

}

] /* range_scan_alternatives */,

“analyzing_roworder_intersect”: {

“usable”: false,

“cause”: “too_few_roworder_scans”

} /* analyzing_roworder_intersect */

} /* analyzing_range_alternatives */,

“chosen_range_access_summary”: {

“range_access_plan”: {

“type”: “range_scan”,

“index”: “PRIMARY”,

“rows”: 3,

“ranges”: [

“id < 4”

] /* ranges */

} /* range_access_plan */,

“rows_for_plan”: 3,

“cost_for_plan”: 1.6154,

“chosen”: true

} /* chosen_range_access_summary */

} /* range_analysis */

}

] /* rows_estimation */

},

{

“considered_execution_plans”: [

{

“plan_prefix”: [

] /* plan_prefix */,

“table”: “tb_item”,

“best_access_path”: {

“considered_access_paths”: [

{

“access_type”: “range”,

“rows”: 3,

“cost”: 2.2154,

“chosen”: true

}

] /* considered_access_paths */

} /* best_access_path */,

“cost_for_plan”: 2.2154,

“rows_for_plan”: 3,

“chosen”: true

}

] /* considered_execution_plans */

},

小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Java工程师,想要提升技能,往往是自己摸索成长,但自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年最新Java开发全套学习资料》送给大家,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
img
img
img

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频

如果你觉得这些内容对你有帮助,可以添加下面V无偿领取!(备注Java)
img

1200页Java架构面试专题及答案

小编整理不易,对这份1200页Java架构面试专题及答案感兴趣劳烦帮忙转发/点赞

百度、字节、美团等大厂常见面试题

/* analyzing_range_alternatives */,

“chosen_range_access_summary”: {

“range_access_plan”: {

“type”: “range_scan”,

“index”: “PRIMARY”,

“rows”: 3,

“ranges”: [

“id < 4”

] /* ranges */

} /* range_access_plan */,

“rows_for_plan”: 3,

“cost_for_plan”: 1.6154,

“chosen”: true

} /* chosen_range_access_summary */

} /* range_analysis */

}

] /* rows_estimation */

},

{

“considered_execution_plans”: [

{

“plan_prefix”: [

] /* plan_prefix */,

“table”: “tb_item”,

“best_access_path”: {

“considered_access_paths”: [

{

“access_type”: “range”,

“rows”: 3,

“cost”: 2.2154,

“chosen”: true

}

] /* considered_access_paths */

} /* best_access_path */,

“cost_for_plan”: 2.2154,

“rows_for_plan”: 3,

“chosen”: true

}

] /* considered_execution_plans */

},

小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Java工程师,想要提升技能,往往是自己摸索成长,但自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年最新Java开发全套学习资料》送给大家,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
[外链图片转存中…(img-FpCDYpXq-1711193933661)]
[外链图片转存中…(img-POGC8Qy3-1711193933662)]
[外链图片转存中…(img-fWtvnBtO-1711193933662)]

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频

如果你觉得这些内容对你有帮助,可以添加下面V无偿领取!(备注Java)
[外链图片转存中…(img-5qdUp0Ui-1711193933663)]

1200页Java架构面试专题及答案

小编整理不易,对这份1200页Java架构面试专题及答案感兴趣劳烦帮忙转发/点赞

[外链图片转存中…(img-pJsyqOzx-1711193933663)]

[外链图片转存中…(img-GBflsnVx-1711193933664)]

百度、字节、美团等大厂常见面试题

[外链图片转存中…(img-1jTXbB5Q-1711193933664)]

本文已被CODING开源项目:【一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码】收录

  • 21
    点赞
  • 25
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值