MySQL高級知識(五)——索引分析


前言:前面已經學習了explain(執行計划)的相關知識,這里利用explain對索引進行優化分析。


0.准備

首先創建三張表:tb_emp(職工表)、tb_dept(部門表)和tb_desc(描述表)

1)tb_emp表。

DROP TABLE IF EXISTS `tb_emp`;
CREATE TABLE `tb_emp` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `username` varchar(20) NOT NULL,
`deptid` int(11) NOT NULL,
PRIMARY KEY (`id`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; -- ---------------------------- INSERT INTO `tb_emp`(username,deptid) VALUES ('Tom', '1'); INSERT INTO `tb_emp`(username,deptid) VALUES ('Jack', '1'); INSERT INTO `tb_emp`(username,deptid) VALUES ('Mary', '2'); INSERT INTO `tb_emp`(username,deptid) VALUES ('Rose', '3');

2)tb_dept表。

DROP TABLE IF EXISTS `tb_dept`;
CREATE TABLE `tb_dept` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `name` varchar(20) NOT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;
-- ----------------------------
INSERT INTO `tb_dept`(name) VALUES ('綜合部');
INSERT INTO `tb_dept`(name) VALUES ('研發');
INSERT INTO `tb_dept`(name) VALUES ('測試');
INSERT INTO `tb_dept`(name) VALUES ('總裁');

3)tb_desc表。

DROP TABLE IF EXISTS `tb_desc`;
CREATE TABLE `tb_desc` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `empid` int(11) DEFAULT NULL,
  `deptid` int(11) DEFAULT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;
-- ----------------------------
INSERT INTO `tb_desc`(empid,deptid) VALUES (1, 1);
INSERT INTO `tb_desc`(empid,deptid) VALUES (2, 1);
INSERT INTO `tb_desc`(empid,deptid) VALUES (3, 2);
INSERT INTO `tb_desc`(empid,deptid) VALUES (4, 3);

注:這里強行將員工表與部門表不直接關聯,通過第三張表(描述表)進行關聯,主要為了進行join的分析。

1.left join

#1.首先執行查詢。

#2.通過explain進行分析。

分析:從explain執行結果可以看出對兩表都是用了全表掃描(ALL),並且在tb_desc表中還使用了join連接緩存,需要進行優化。但是如何優化?是在左表建立索引還是右表建立索引呢?因為左連接左表是全有,所以應該在右表建立索引。

#3.右表創建索引。

通過explain執行可以看到,在創建索引后,獲得了比較不錯的結果。(type=ref,Extra=Using index)。

結論:left join(左連接)情況下,應該在右表(tb_desc)創建索引。

2.right join

通過上面left join的例子,我們直接交換兩表位置,並將left join改變成right join。

分析:

與left join進行對比,可以得到如下結論:

#1.在left join下,首先執行tb_emp(左表),type=ALL,因為左連接情況下左表全有,因此我們在tb_desc(右表)創建索引,得到比較理想的效果。

#2.在right join下(我們交換了tb_emp和tb_desc的位置),執行順序:tb_emp(右表)→ tb_desc(左表)。右表type=ALL,因為右連接情況下右表全有,因此在左表(tb_desc,我們交換了位置)創建索引,效果肯定和left join一樣。

總結

left join(左連接):右表創建索引。

right join(右連接):左表創建索引。

簡記:左右外連接,索引相反建(left:右表建,right:左表建)。


by Shawn Chen,2018.6.23日,晚。


相關內容

MySQL高級知識系列目錄


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM