成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

數據庫實踐丨MySQL多表join分析

數據庫 MySQL
在數據庫查詢中,往往會需要查詢多個表的數據,比如查詢會員信息同時查詢關于這個會員的訂單信息,如果分語句查詢的話,效率會很低,就需要用到join關鍵字來連表查詢了。

Join并行

Join并行1. 多表join介紹2. 多表Join的方式不使用Join buffer使用Join buffer3. Join執行流程(老執行器)

1. 多表join介紹

JOIN子句用于根據兩個或多個表之間的相關列來組合它們。 例如:

Orders:

Customers:

 

  1. SELECT Orders.OrderID, Customers.CustomerName, Orders.OrderDate  
  2. FROM Orders  
  3. INNER JOIN Customers ON Orders.CustomerID=Customers.CustomerID; 

2. 多表Join的方式

Hash join使用新執行器實現,在這里不做討論

MySQL支持的都是Nested-Loop Join,以及它的變種。

不使用Join buffer

a) Simple Nested-Loop

對r表的每一行,完整掃描s表,根據r[i]-s[i]組成的行去判斷是否滿足條件,并返回滿足條件的結果給客戶端。

 

  1. mysql> show create table t1;  
  2. +-------+----------------------------------------------------------------------------------------------------------------+  
  3. | Table | Create Table                                                                                                   |  
  4. +-------+----------------------------------------------------------------------------------------------------------------+  
  5. | t1    | CREATE TABLE `t1` (  
  6.  `id` int(11) NOT NULL  
  7. ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci |  
  8. +-------+----------------------------------------------------------------------------------------------------------------+  
  9. 1 row in set (0.00 sec)  
  10. mysql> show create table t3;  
  11. +-------+--------------------------------------------------------------------------------------------------------------------+  
  12. | Table | Create Table                                                                                                       |  
  13. +-------+--------------------------------------------------------------------------------------------------------------------+  
  14. | t3    | CREATE TABLE `t3` (  
  15.  `id` int(11) DEFAULT NULL  
  16. ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci |  
  17. +-------+--------------------------------------------------------------------------------------------------------------------+  
  18. 1 row in set (0.00 sec)  
  19. mysql> explain select /*+ NO_BNL() */ * from t1, t3 where t1.id = t3.id;  
  20. +----+-------------+-------+------------+------+---------------+------+---------+------+------+----------+-------------+  
  21. | id | select_type | table | partitions | type | possible_keys | key  | key_len | ref  | rows | filtered | Extra       |  
  22. +----+-------------+-------+------------+------+---------------+------+---------+------+------+----------+-------------+  
  23. |  1 | SIMPLE      | t1    | NULL       | ALL  | NULL          | NULL | NULL    | NULL |    2 |   100.00 | NULL        |  
  24. |  1 | SIMPLE      | t3    | NULL       | ALL  | NULL          | NULL | NULL    | NULL |    2 |    50.00 | Using where |  
  25. +----+-------------+-------+------------+------+---------------+------+---------+------+------+----------+-------------+  
  26. 2 rows in set, 1 warning (0.00 sec) 

b) Index Nested-Loop

對r表的每一行,先根據連接條件去查詢s表索引,然后回表查到匹配的數據,并返回滿足條件的結果給客戶端。

 

  1. mysql> show create table t2;  
  2. +-------+---------------------------------------------------------------------------------------------------------------------------------------+  
  3. | Table | Create Table                                                                                                                          |  
  4. +-------+---------------------------------------------------------------------------------------------------------------------------------------+  
  5. | t2    | CREATE TABLE `t2` (  
  6.  `id` int(11) NOT NULL,  
  7.  KEY `index1` (`id`)  
  8. ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci |  
  9. +-------+---------------------------------------------------------------------------------------------------------------------------------------+  
  10. 1 row in set (0.00 sec)  
  11. mysql> explain select * from t1, t2 where t1.id = t2.id;  
  12. +----+-------------+-------+------------+------+---------------+--------+---------+------------+------+----------+-------------+  
  13. | id | select_type | table | partitions | type | possible_keys | key    | key_len | ref        | rows | filtered | Extra       |  
  14. +----+-------------+-------+------------+------+---------------+--------+---------+------------+------+----------+-------------+  
  15. |  1 | SIMPLE      | t1    | NULL       | ALL  | NULL          | NULL   | NULL    | NULL       |    2 |   100.00 | NULL        |  
  16. |  1 | SIMPLE      | t2    | NULL       | ref  | index1        | index1 | 4       | test.t1.id |    1 |   100.00 | Using index |  
  17. +----+-------------+-------+------------+------+---------------+--------+---------+------------+------+----------+-------------+  
  18. 2 rows in set, 1 warning (0.00 sec)  

使用Join buffer

a) Block Nested Loop

從r表讀取一部分數據到join cache中,當r表數據讀完或者join cache滿后,做join操作。 

  1. JOIN_CACHE_BNL::join_matching_records(){  
  2.  do {  
  3.    //讀取s表的每一行 
  4.     qep_tab->table()->file->position(qep_tab->table()->record[0]);  
  5.    //針對s的每一行,遍歷join buffer  
  6.    for(each record in join buffer) {  
  7.      get_record();  
  8.      rc = generate_full_extensions(get_curr_rec());  
  9.      //如果不符合條件,直接返回  
  10.      if (rc != NESTED_LOOP_OK) return rc;  
  11.    }  
  12.  } while(!(error = iterator->Read()))  
  13.  
  1. mysql> explain select  * from t1, t3 where t1.id = t3.id;  
  2. +----+-------------+-------+------------+------+---------------+------+---------+------+------+----------+----------------------------------------------------+  
  3. | id | select_type | table | partitions | type | possible_keys | key  | key_len | ref  | rows | filtered | Extra                                              |  
  4. +----+-------------+-------+------------+------+---------------+------+---------+------+------+----------+----------------------------------------------------+  
  5. |  1 | SIMPLE      | t1    | NULL       | ALL  | NULL          | NULL | NULL    | NULL |    2 |   100.00 | NULL                                               |  
  6. |  1 | SIMPLE      | t3    | NULL       | ALL  | NULL          | NULL | NULL    | NULL |    2 |    50.00 | Using where; Using join buffer (Block Nested Loop) |  
  7. +----+-------------+-------+------------+------+---------------+------+---------+------+------+----------+----------------------------------------------------+  
  8. 2 rows in set, 1 warning (0.00 sec) 

 b) Batched Key Access

從r表讀取一部分數據到join cache中,s表中記錄r表被連接的列的值作為索引,查詢所有符合條件的索引,然后將這些符合條件的索引排序,然后統一回表查詢記錄。

其中,對于每一個cached record,都會有一個key,通過這個key去s表掃描所需的數據。 

  1. dsmrr_fill_buffer(){  
  2.  while((rowids_buf_cur < rowids_buf_end) &&  
  3.        !(res = h2->handler::multi_range_read_next(&range_info))){  
  4.    //下壓的index條件  
  5.    if (h2->mrr_funcs.skip_index_tuple &&  
  6.        h2->mrr_funcs.skip_index_tuple(h2->mrr_iter, curr_range->ptr))  
  7.      continue;  
  8.    memcpy(rowids_buf_cur, h2->ref, h2->ref_length);  
  9.  }  
  10.  varlen_sort(  
  11.      rowids_buf, rowids_buf_cur, elem_size,  
  12.      [this](const uchar *a, const uchar *b) { return h->cmp_ref(a, b) < 0; });  
  13.  
  14. dsmrr_next(){  
  15.  do{  
  16.    if (rowids_buf_cur == rowids_buf_last) {  
  17.      dsmrr_fill_buffer();  
  18.    }  
  19.    // first match  
  20.    if (h2->mrr_funcs.skip_record &&  
  21.        h2->mrr_funcs.skip_record(h2->mrr_iter, (char *)cur_range_info, rowid))  
  22.      continue;  
  23.    res = h->ha_rnd_pos(table->record[0], rowid);  
  24.    break;  
  25.  } while(true);  
  26.  
  27. JOIN_CACHE_BKA::join_matching_records(){  
  28.  while (!(error = file->ha_multi_range_read_next((char **)&rec_ptr))) {  
  29.    get_record_by_pos(rec_ptr);  
  30.    rc = generate_full_extensions(rec_ptr);  
  31.      if (rc != NESTED_LOOP_OK) return rc;  
  32.  }  
  33.  
  1. mysql> show create table t1;  
  2. +-------+-------------------------------------------------------------------------------------------------------------------------------------------------+  
  3. | Table | Create Table                                                                                                                                    |  
  4. +-------+-------------------------------------------------------------------------------------------------------------------------------------------------+  
  5. | t1    | CREATE TABLE `t1` (  
  6.  `f1` int(11) DEFAULT NULL, 
  7.  `f2` int(11) DEFAULT NULL  
  8. ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci |  
  9. +-------+-------------------------------------------------------------------------------------------------------------------------------------------------+  
  10. 1 row in set (0.00 sec)  
  11. mysql> show create table t2;  
  12. +-------+------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------+ 
  13. | Table | Create Table                                                                                                                                                                                | 
  14. +-------+------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------+ 
  15. | t2    | CREATE TABLE `t2` (  
  16.  `f1` int(11) NOT NULL,  
  17.  `f2` int(11) NOT NULL,  
  18.  `f3` char(200) DEFAULT NULL,  
  19.  KEY `f1` (`f1`,`f2`)  
  20. ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci |  
  21. +-------+------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------+ 
  22. 1 row in set (0.00 sec)  
  23. mysql> explain SELECT /*+ BKA() */ t2.f1, t2.f2, t2.f3 FROM t1,t2 WHERE t1.f1=t2.f1 AND t2.f2 BETWEEN t1.f1 and t1.f2 and t2.f2 + 1 >= t1.f1 + 1;  
  24. +----+-------------+-------+------------+------+---------------+------+---------+-------------+------+----------+---------------------------------------------------------------+ 
  25. | id | select_type | table | partitions | type | possible_keys | key  | key_len | ref         | rows | filtered | Extra                                                      | 
  26. +----+-------------+-------+------------+------+---------------+------+---------+-------------+------+----------+---------------------------------------------------------------+ 
  27. |  1 | SIMPLE      | t1    | NULL       | ALL  | NULL          | NULL | NULL    | NULL        |    3 |   100.00 | Using where                                                   | 
  28. |  1 | SIMPLE      | t2    | NULL       | ref  | f1            | f1   | 4       | test1.t1.f1 |    7 |    11.11 | Using index condition; Using join buffer (Batched Key Access) | 
  29. +----+-------------+-------+------------+------+---------------+------+---------+-------------+------+----------+---------------------------------------------------------------+ 
  30. 2 rows in set, 1 warning (0.00 sec) 

c) Batched Key Access(unique)

與Batched Key Access不同的是,r中的列是s的唯一索引,在r記錄寫入join cache的時候,會記錄一個key的hash table,僅針對不同的key去s表中查詢。(疑問,為什么只有unique的時候才能用這種方式?不是unique的話,s表中可能會掃描出多條數據,也可以用這種方式去處理,減少s表的重復掃描)。 

  1. JOIN_CACHE_BKA_UNIQUE::join_matching_records(){  
  2.  while (!(error = file->ha_multi_range_read_next((char **)&key_chain_ptr))) {  
  3.    do(each record in chain){  
  4.      get_record_by_pos(rec_ptr);  
  5.      rc = generate_full_extensions(rec_ptr);  
  6.        if (rc != NESTED_LOOP_OK) return rc;  
  7.      }  
  8.  }  
  9.  
  1. mysql> show create table city;  
  2. +-------+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------+ 
  3. | Table | Create Table                                                                                                                                                                                                                                                                                                                                       | 
  4. +-------+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------+ 
  5. | city  | CREATE TABLE `city` (  
  6.  `ID` int(11) NOT NULL AUTO_INCREMENT,  
  7.  `Name` char(35) NOT NULL DEFAULT '',  
  8.  `Country` char(3) NOT NULL DEFAULT '',  
  9.  `Population` int(11) NOT NULL DEFAULT '0',  
  10.  PRIMARY KEY (`ID`),  
  11.  KEY `Population` (`Population`),  
  12.  KEY `Country` (`Country`)  
  13. ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci |  
  14. +-------+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------+ 
  15. 1 row in set (0.00 sec)  
  16. mysql> show create table country;  
  17. +---------+-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------+ 
  18. | Table   | Create Table                                                                                                                                                                                                                                                                                                                                                   | 
  19. +---------+-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------+ 
  20. | country | CREATE TABLE `country` (  
  21.  `Code` char(3) NOT NULL DEFAULT '',  
  22.  `Name` char(52) NOT NULL DEFAULT '',  
  23.  `SurfaceArea` float(10,2) NOT NULL DEFAULT '0.00',  
  24.  `Population` int(11) NOT NULL DEFAULT '0',  
  25.  `Capital` int(11) DEFAULT NULL,  
  26.  PRIMARY KEY (`Code`),  
  27.  UNIQUE KEY `Name` (`Name`)  
  28. ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci |  
  29. +---------+-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------+ 
  30. 1 row in set (0.01 sec)  
  31. mysql> EXPLAIN SELECT city.Name, country.Name FROM city,country WHERE city.country=country.Code AND  country.Name LIKE 'L%' AND city.Population > 100000; 
  32. +----+-------------+---------+------------+-------+--------------------+---------+---------+--------------------+------+----------+--------------------------------------------------------------+ 
  33. | id | select_type | table   | partitions | type  | possible_keys      | key     | key_len | ref                | rows | filtered | Extra                                                        | 
  34. +----+-------------+---------+------------+-------+--------------------+---------+---------+--------------------+------+----------+--------------------------------------------------------------+ 
  35. |  1 | SIMPLE      | country | NULL       | index | PRIMARY,Name       | Name    | 208     | NULL               |    1 |   100.00 | Using where; Using index                                     | 
  36. |  1 | SIMPLE      | city    | NULL       | ref   | Population,Country | Country | 12      | test1.country.Code |    1 |   100.00 | Using where; Using join buffer (Batched Key Access (unique)) | 
  37. +----+-------------+---------+------------+-------+--------------------+---------+---------+--------------------+------+----------+--------------------------------------------------------------+ 
  38. 2 rows in set, 1 warning (0.01 sec) 

 3. Join執行流程(老執行器)

 

  1. sub_select <-------------------------------------------- 
  2.  | -> iterator::read() // 讀一行數據                    |  
  3.  | -> evaluate_join_record()  //檢查這行數據是否符合條件 |  
  4.  | -> next_select() ---+                               |  
  5.                        |                               |  
  6. sub_select_op  <--------+                               |  
  7.  | -> op->put_record() // 前表數據寫入join cache        |  
  8.    | -> put_record_in_cache()                          |  
  9.    | -> join->record()                                 |  
  10.      | -> join_matching_records()                      |  
  11.        | -> (qep_tab->next_select)(join, qep_tab + 1, 0) // 繼續調用next_select  
  12.    | -> end_send()  

 

責任編輯:龐桂玉 來源: segmentfault
相關推薦

2022-09-05 10:06:21

MySQL外循環內循環

2018-07-30 15:00:05

數據庫MySQLJOIN

2017-07-18 17:07:40

數據庫 MyCATJoin

2011-07-06 14:12:20

MySQLPercona

2011-07-06 10:49:50

MySQL優化

2010-05-21 14:01:23

MySQL數據庫

2010-06-07 16:22:55

MySQL數據庫

2010-06-02 18:07:44

MySQL數據庫

2018-06-26 15:58:06

數據庫MySQL索引優化

2021-04-09 08:21:25

數據庫索引數據

2017-06-22 16:00:07

數據庫NoSQL遷移實踐

2022-02-10 10:51:35

數據庫

2024-05-28 00:00:30

Golang數據庫

2013-10-08 09:54:41

數據庫安全數據庫管理

2023-09-12 09:45:54

Java數據庫

2017-11-22 09:20:41

數據庫在線數據遷移Subscriptio

2010-11-30 11:26:49

2017-03-15 15:14:03

MySQL數據庫高可用性

2025-04-08 06:00:00

點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 天堂久 | 欧美福利在线 | 国产一级视频在线观看 | 日韩精品专区在线影院重磅 | 中文字幕久久久 | 亚洲三区在线观看 | 18成人在线观看 | 超黄视频网站 | 99精品欧美一区二区三区 | www亚洲精品| 午夜视频在线观看网站 | 中文字幕视频一区 | 四虎影院欧美 | 欧美亚洲成人网 | 久久精品综合 | 国产精品乱码一区二区三区 | 中文字幕在线一区二区三区 | 精品国产一区二区三区av片 | 国内久久 | 久久三级影院 | 国产成人精品午夜 | 国产性色视频 | 中文字幕亚洲无线 | 亚洲 欧美 在线 一区 | 国产成人a亚洲精品 | 国产精品视频一二三区 | 99热这里有精品 | 91高清在线视频 | 影视一区 | 成人欧美一区二区三区黑人孕妇 | 中文在线播放 | 秋霞电影一区二区 | 免费久久精品视频 | 91黄色免费看 | 亚洲网站在线观看 | 欧美在线免费 | 中文字幕一区二区三区四区不卡 | 亚洲精品国产a久久久久久 午夜影院网站 | 国产精品人人做人人爽 | 午夜色婷婷 | 亚洲一区二区免费 |