电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

您的位置:首頁技術文章
文章詳情頁

詳解MySQL 聯合查詢優化機制

瀏覽:2日期:2023-10-03 12:56:25
MySQL 聯合查詢執行策略。

以一個 UNION 查詢為例,MySQL 執行 UNION 查詢時,會把他們當做一系列的單個查詢語句,然后把對應的結果放入到臨時表中,最終再讀出來返回。在 MySQL中,每個獨立的查詢都是一個聯合查詢,從臨時表讀取返回結果也一樣。

這種情形下,MySQL 的聯合查詢執行很簡單——它將這里的聯合查詢當做是嵌套循環的聯合查詢。這意味著 MySQL 會運行一個循環去從數據表讀取數據行,然而在運行一個嵌套循環從下一個表讀取匹配的數據行。這個過程一直持續,直到找到聯合查詢中的所有匹配的數據行。然后再根據 SELECT 語句中需要的列去構建返回結果。如下面的查詢語句所示:

SELECT tb1.col1, tb2.col2FROM tb1 INNER JOIN tb2 USING(col3)WHERE tb1.col1 IN(5,6);

實際轉換為 MySQL可能執行的偽代碼是下面這樣的:

outer_iter = iterator over tb1 where col1 IN(5,6);outer_row = outer_iter.next;while outer_rowinner_iter = iterator over tb2 where col3 = outer_row.col3;inner_row = inner_iter.next while inner_row output [outer_row.col1, inner_row.col2];inner_row = inner_iter.next;end outer_row = outer.iter.next;end

轉換為偽代碼后如下所示

outer_iter = iterator over tb1 where col1 IN(5,6);outer_row = outer_iter.next;while outer_rowinner_iter = iterator over tb2 where col3 = outer_row.col3;inner_row = inner_iter.next if inner_rowwhile inner_row output [outer_row.col1, inner_row.col2]; inner_row = inner_iter.next;end else output [outer_row.col1, NULL];end outer_row = outer.iter.next;end

另一個方式可視化展現查詢計劃的方式是使用泳道圖的形式。下面的圖展示了 內連接查詢的泳道圖。

詳解MySQL 聯合查詢優化機制

MySQL 執行的各類查詢基本上都是相同的方式。例如,在 FROM 條件里需要先執行的子查詢時,也是先將結果放入臨時表,然后再把臨時表當作普通表后聯合來處理。MySQL 執行聯合查詢時也是使用臨時表,然后將右連接查詢重寫為等價的左連接。簡而言之,當前版本的 MySQL 會盡可能把各類查詢轉成這種方式處理(最新版本 MySQL5.6以后引入了更多的復雜的處理方式)。

當然,并不是所有合法的 SQL 查詢語句都可以這么做,有些查詢這么做的效果可能很差。

執行計劃

MySQL不像其他很多數據庫產品,它不會將查詢語句產生字節碼去執行查詢計劃。實際上,查詢執行計劃是一棵指令樹,查詢執行引擎根據這棵樹產生查詢結果。最終的查詢計劃包含了足夠多的信息去重構最初的查詢。如果在查詢語句上執行EXPLAIN EXTENDED(MySQL 8以后不需要加 EXTENDED),然后再執行SHOW WARNINGS,就可以看到重構后的查詢。

詳解MySQL 聯合查詢優化機制

對于多表查詢在概念上可以用樹代表。例如,一個4張表的查詢可能長得像下面的樹一樣。這在計算機里稱為平衡樹,

然而這不是 MySQL 執行查詢的方式。如前所述,MySQL 總是從一張數據表開始,然后再從下一張表尋找匹配的數據行。因此,MySQL 的查詢計劃看起來像下面的左深連接樹。

詳解MySQL 聯合查詢優化機制

聯合查詢優化器

MySQL 的查詢優化器中最重要的部分是聯合查詢優化器,由它來決定多表查詢執行過程的最優順序。通常可以通過多種聯合查詢的次序獲取相同的結果。聯合查詢優化器試圖估計這些方案的代價,然后選擇最低代價的方案去執行。

下面是一個查詢相同結果,但不同次序的聯合查詢示例。

SELECT film.film_id, film.title, film.release_year, actor.actor_id, actor.first_name, actor.last_nameFROM sakila.filmINNER JOIN sakila.film_actor USING(film_id)INNER JOIN sakila.actor USING(actor_id);

這里面可能會有一些不同的查詢方式。比如,MySQL 可以從 film 表開始,使用 film_actor 的film_id 索引去查找對應的 actor_di 值,然后再從 actor 表使用主鍵找到對應的 actor 數據行。而 Oracle 用戶可能會表述為:“film 表是 film_actor 的驅動表,而 film_actor 是 actor 表的驅動表”。而使用 Explain 解析的結果如下:

******** 1.row ********id: 1select_type: SIMPLEtable: actortype: ALLpossible_keys: PRIMARYkey: NULLkey_len: NULLref: NULLrows: 200Extra:******** 2.row ********id: 1select_type: SIMPLEtable: film_actortype: refpossible_keys: PRIMARY, idx_fk_film_idkey: PRIMARYkey_len: 2ref: sakila.film.film_idrows: 1Extra: USING index******** 3.row ********id: 1select_type: SIMPLEtable: filmtype: eq_refpossible_keys: PRIMARYkey: PRIMARYkey_len: 2ref: sakila.film_actor.film_idrows: 1Extra:

這個執行計劃與我們猜想的有很大不同。MySQL 首先從 actor 表開始,然后次序是反向的。這是否真的更有效?我們可以在 EXPLAIN 上加上 STRAIGHT_JOIN 來避免優化:

EXPLAIN SELECT STRAIGHT_JOIN film.film_id, film.title, film.release_year, actor.actor_id, actor.first_name, actor.last_nameFROM sakila.filmINNER JOIN sakila.film_actor USING(film_id)INNER JOIN sakila.actor USING(actor_id);

******** 1.row ********id: 1select_type: SIMPLEtable: filmtype: ALLpossible_keys: PRIMARYkey: NULLkey_len: NULLref: NULLrows: 951Extra:******** 2.row ********id: 1select_type: SIMPLEtable: film_actortype: refpossible_keys: PRIMARY, idx_fk_film_idkey: idx_fk_film_idkey_len: 2ref: sakila.film.film_idrows: 1Extra: USING index******** 3.row ********id: 1select_type: SIMPLEtable: actortype: eq_refpossible_keys: PRIMARYkey: PRIMARYkey_len: 2ref: sakila.film_actor.actor_idrows: 1Extra:

這解釋了為什么MySQL 為什么需要反序執行查詢,這會使得檢查的數據行更少。

先查詢 film 表會需要對 film_actor 和 actor 進行951次查詢(最外層循環) 如果將 actor表前置,則只需要對其他表進行200次查詢。

從這個例子可以看出,MySQL 的聯合查詢優化器可以通過調整查詢表次序降低查詢代價。重新排序后的聯合查詢通常是很有效的優化,通常是幾倍性能的提高。如果沒有性能提高的話,也可以使用 STRAIGHT_JOIN 來避免重排序,而使用我們自己認為最好的查詢方式。這種情況實際遇到的會很少,大部分情況下,聯合查詢優化器都會比人做得更出色。

聯合查詢優化器視圖以最低完成代價構建一個查詢執行樹。如果有可能,它會從全部的單表計劃開始,檢查所有可能的子樹組合。不幸的是,一個 N 張表的聯合查詢會有 N 個階乘的組合次序數量。這被稱之為所有可能的查詢計劃的搜索空間,這個數量增長非常快。一個10張表的聯合索引會有3628800個不同的方式!一旦搜索空間增長到過大,會導致查詢的優化十分久,這時候服務端會停止做全量分析,替代以類似貪婪算法的方式完成優化。這個數量通過 optimizer_search_depth 系統變量控制,可以自己修改該參數。

標簽: MySQL 數據庫
相關文章:
主站蜘蛛池模板: 锂电混合机-新能源混合机-正极材料混料机-高镍,三元材料混料机-负极,包覆混合机-贝尔专业混合混料搅拌机械系统设备厂家 | 兰州牛肉面加盟,兰州牛肉拉面加盟-京穆兰牛肉面 | 污水提升器,污水提升泵,污水提升装置-德国泽德(zehnder)水泵系统有限公司 | 流量检测仪-气密性检测装置-密封性试验仪-东莞市奥图自动化科技有限公司 | 武汉天安盾电子设备有限公司 - 安盾安检,武汉安检门,武汉安检机,武汉金属探测器,武汉测温安检门,武汉X光行李安检机,武汉防爆罐,武汉车底安全检查,武汉液体探测仪,武汉安检防爆设备 | 地图标注-手机导航电子地图如何标注-房地产商场地图标记【DiTuBiaoZhu.net】 | 安平县鑫川金属丝网制品有限公司,声屏障,高速声屏障,百叶孔声屏障,大弧形声屏障,凹凸穿孔声屏障,铁路声屏障,顶部弧形声屏障,玻璃钢吸音板 | 全自动过滤器_反冲洗过滤器_自清洗过滤器_量子除垢环_量子环除垢_量子除垢 - 安士睿(北京)过滤设备有限公司 | 磁棒电感生产厂家-电感器厂家-电感定制-贴片功率电感供应商-棒形电感生产厂家-苏州谷景电子有限公司 | UV固化机_UVLED光固化机_UV干燥机生产厂家-上海冠顶公司专业生产UV固化机设备 | 纸箱抗压机,拉力机,脂肪测定仪,定氮仪-山东德瑞克仪器有限公司 | 真空吸污车_高压清洗车厂家-程力专用汽车股份有限公司官网 | 制氮设备_PSA制氮机_激光切割制氮机_氮气机生产厂家-苏州西斯气体设备有限公司 | 宁夏档案密集柜,智能密集柜,电动手摇密集柜-盛隆柜业宁夏档案密集柜厂家 | 304不锈钢无缝管_不锈钢管厂家 - 隆达钢业集团有限公司 | 无尘烘箱_洁净烤箱_真空无氧烤箱_半导体烤箱_电子防潮柜-深圳市怡和兴机电 | 焦作网 WWW.JZRB.COM | 重庆小面培训_重庆小面技术培训学习班哪家好【终身免费复学】 | 电动葫芦|手拉葫芦|环链电动葫芦|微型电动葫芦-北京市凌鹰起重机械有限公司 | 造价工程师网,考试时间查询,报名入口信息-网站首页 | 活性氧化铝|无烟煤滤料|活性氧化铝厂家|锰砂滤料厂家-河南新泰净水材料有限公司 | 世纪豪门官网 世纪豪门集成吊顶加盟电话 世纪豪门售后电话 | 祝融环境-地源热泵多恒系统高新技术企业,舒适生活环境缔造者! | 丝印油墨_水性油墨_环保油墨油漆厂家_37国际化工 | 好看的韩国漫画_韩漫在线免费阅读-汗汗漫画 | 上海璟文空运首页_一级航空货运代理公司_机场快递当日达 | 纯水电导率测定仪-万用气体检测仪-低钠测定仪-米沃奇科技(北京)有限公司www.milwaukeeinst.cn 锂辉石检测仪器,水泥成分快速分析仪-湘潭宇科分析仪器有限公司 手术室净化装修-手术室净化工程公司-华锐手术室净化厂家 | 吸音板,隔音板,吸音材料,吸音板价格,声学材料 - 佛山诺声吸音板厂家 | 温州在线网| 防爆正压柜厂家_防爆配电箱_防爆控制箱_防爆空调_-盛通防爆 | 高清视频编码器,4K音视频编解码器,直播编码器,流媒体服务器,深圳海威视讯技术有限公司 | 杭州中策电线|中策电缆|中策电线|杭州中策电缆|杭州中策电缆永通集团有限公司 | 杜康白酒加盟_杜康酒代理_杜康酒招商加盟官网_杜康酒厂加盟总代理—杜康酒神全国运营中心 | 软文推广发布平台_新闻稿件自助发布_媒体邀约-澜媒宝 | 成都APP开发-成都App定制-成都app开发公司-【未来久】 | 建大仁科-温湿度变送器|温湿度传感器|温湿度记录仪_厂家_价格-山东仁科 | 语料库-提供经典范文,文案句子,常用文书,您的写作得力助手 | 苏州工作服定做-工作服定制-工作服厂家网站-尺品服饰科技(苏州)有限公司 | 合肥钣金加工-安徽激光切割加工-机箱机柜加工厂家-合肥通快 | 防爆正压柜厂家_防爆配电箱_防爆控制箱_防爆空调_-盛通防爆 | 河南膏药贴牌-膏药代加工-膏药oem厂家-洛阳今世康医药科技有限公司 |