7個工程應用中資料庫性能優化經驗分享

来源:https://www.cnblogs.com/huaweiyun/archive/2023/05/26/17434912.html
-Advertisement-
Play Games

摘要:此篇文章分別從sql執行過程、執行計劃、索引數據結構、索引查詢提速原理、聚焦索引、左首碼優化原則、自增主鍵索引這些角度談一談我們對資料庫優化的理解。 本文分享自華為雲社區《工程應用中資料庫性能優化經驗小結》,作者: 葉工 。 1、前言 現階段交付的演算法產品,絕大多數涉及到資料庫的使用。它承載的 ...


摘要:此篇文章分別從sql執行過程、執行計劃、索引數據結構、索引查詢提速原理、聚焦索引、左首碼優化原則、自增主鍵索引這些角度談一談我們對資料庫優化的理解。

本文分享自華為雲社區《工程應用中資料庫性能優化經驗小結》,作者: 葉工 。

1、前言

現階段交付的演算法產品,絕大多數涉及到資料庫的使用。它承載的內容包括:用戶許可權管理、數據集信息、非同步推論的結果、個性化配置等等。

在OCR場景下,數據集體量通常較大(一個數據集幾十萬張圖片),而資料庫往往部署在客戶共用資料庫中(同時運行大量其他業務),甚至只能和演算法鏡像共用同一臺伺服器,因此在後臺研發中尤其要關心資料庫性能瓶頸。

此篇文章分別從 sql執行過程、執行計劃、索引數據結構、索引查詢提速原理、聚焦索引、左首碼優化原則、自增主鍵索引這些角度談一談我們對資料庫優化的理解。

2、ORM場景下如何獲得完整SQL語句

1. 線上環境可以通過連接池進行慢SQL攔截,併發出告警通知

2. 測試階段,因為使用預編譯語句或ORM框架,無法獲取完整SQL時可以使用資料庫日誌方式獲取

set global general_log=on;
show variables where Variable_name="general_log_file"

2.1 SQL執行過程

分析器:分析SQL,需要使用哪些表,使用哪些條件(知道要乾什麼)

優化器: 對各種執行過程做性能評估,挑選代價最小的執行過程,代價只是優化器認為的,不一定正確 (怎麼樣做最快)

執行器:調用引擎介面,返回數據,引擎是插件式,類似編程時多態,在創建表時可以選擇相應的存儲 引擎

2.2 執行計劃

SQL前加explain關鍵詞可以得到SQL的執行計劃,根據執行計劃可以判斷執行過程是否符合預期

explain
SELECT
 db_dataset.uuid AS db_dataset_uuid,
  db_dataset.NAME AS db_dataset_name,
 db_dataset.updated_at AS db_dataset_updated_at,
 db_dataset.created_at AS db_dataset_created_at,
 db_dataset.volume_dir AS db_dataset_volume_dir,
 db_dataset.max_data_count AS db_dataset_max_data_count,
 db_dataset.description AS db_dataset_description
FROM
 db_dataset
  LEFT OUTER JOIN db_manifest ON db_manifest.dataset_id = db_dataset.id AND
 db_manifest.dataset_version = 'annotation_v0'
  LEFT OUTER JOIN db_ai_data ON db_manifest.id = db_ai_data.manifest_id AND
 db_ai_data.deleted = '0'
WHERE
 db_dataset.deleted = 0
GROUP BY
  db_dataset.id

執行計劃反饋列的解釋:

select_type詳解:

type詳解:

查詢使用了何種類型,它在 SQL優化中是一個非常重要的指標,以下性能從好到壞依次是:
system > const > eq_ref > ref > ref_or_null > index_merge > unique_subquery >
index_subquery > range > index > ALL

system :當表僅有一行記錄時(系統表),數據量很少,往往不需要進行磁碟 IO,速度非常 快。

const :表示查詢時命中 primary key 主鍵或者 unique 唯一索引,或者被連接的部分是一個常量 (const)值。這類掃描效率極高,返回數據量少,速度非常快。

eq_ref :查詢時命中主鍵 primary key 或者 unique key 索引, type 就是 eq_ref。

ref :區別於 eq_ref,ref 表示使用非唯一性索引,會找到很多個符合條件的行。

ref_or_null :這種連接類型類似於 ref,區別在於 MySQL 會額外搜索包含 NULL 值的行。

index_merge :使用了索引合併優化方法,一個查詢使用了兩個以上的索引。

EXPLAIN SELECT * FROM user_robot_relate WHERE id > 1 AND user_id = 2;

unique_subquery :替換下麵的 IN 子查詢,子查詢返回不重覆的集合。

value IN (SELECT primary_key FROM single_table WHERE some_expr)

index_subquery :區別於 unique_subquery,用於非唯一索引,可以返回重覆值。

value IN (SELECT key_column FROM single_table WHERE some_expr)

range :使用索引選擇行,僅檢索給定範圍內的行。簡單點說就是針對一個有索引的欄位,給定 範圍檢索數據。在 where 語句中使用 bettween...and、<、>、<=、in 等條件查詢 type 都是 range。 從結果中看到只有對設置了索引的欄位,做範圍檢索 type 才是 range。

EXPLAIN SELECT * FROM user_robot_relate WHERE id BETWEEN 2 AND 3;

index :Index 與 ALL 其實都是讀全表,區別在於 index 是遍歷索引樹讀取,而 ALL 是從硬碟中 讀取。

ALL :將遍歷全表以找到匹配的行,性能最差。

Extra :不適合在其他列中顯示的信息,Explain 中的很多額外的信息會在 Extra 欄位顯示。

Using index:我們在相應的 select 操作中使用了覆蓋索引,通俗一點講就是查詢的列被索引覆蓋,使 用到覆蓋索引查詢速度會非常快,SQL 優化中理想的狀態。

Using where:查詢時未找到可用的索引,進而通過 where 條件過濾獲取所需數據,但要註意的是並不 是所有帶 where 語句的查詢都會顯示 Using where。

Using temporary:表示查詢後結果需要使用臨時表來存儲,一般在排序或者分組查詢時用到。

Using filesort:表示無法利用索引完成的排序操作,也就是 ORDER BY 的欄位沒有索引,通常這樣的 SQL 都是需要優化的。

Using join buffer:在我們聯表查詢的時候,如果表的連接條件沒有用到索引,需要有一個連接緩衝區 來存儲中間結果。

2.3 索引

索引時幫助MySQL高效獲取數據的排好序的數據結構

索引數據結構:

二叉樹

紅黑樹

HashTable

B-Tree

一般不用二叉樹的原因:有序數據將退化成鏈表,深度不可控,如下圖所示

通常也不能用紅黑樹的原因:雖然壓縮了深度,但深度還是不可控,海量數據查找複雜度極高

Hash表:僅支持IN查新,不支持RANGE查詢。使用hash演算法將內容進行hash處理 hash(aaaa) = 2 hash(bbbb) = 2 hash(cccc) = 4

B+ 樹:主流的索引結構

查找過程:

1. 讀取根節點所有元素,因為是有序的,可以利用二分查找,高效查找到指定區間

2. 根據指定區間文件地址找到二級節點,讀取所有元素。

3. 找到葉子節點中指定元素位置。

2.4 索引查詢提速原理

以B+樹索引為例,

如果要查找數據項目29

1、首先進入1號塊,1號塊數據載入如記憶體,發生一次I/O

2、在記憶體中進行二分查找,發現29在17和35之前,於是鎖定P2指針,將3號塊數據載入到記憶體,又發生一次I/O

3、同理在3號塊中走P2指針鎖定8號數據塊,將8號數據塊載入到記憶體,最後發生一次I/O

4、遍歷8號塊的數據就能找到29號數據

如果沒有索引,最壞的情況是整個表格的數據塊都需要載入到記憶體,然後遍歷出結果,將產生大量的I/O開銷和對整表數據的遍歷。

2.5 聚焦索引

聚焦索引尤其適合需要進行RANGE查找的列,這是因為他的葉子節點存放的是有序的數據行。在查詢過程中可以根據WHERE的條件定位到兩端葉子節點,然後將他們之間的整個鏈表結構取出。

2.6 左首碼優化原則

工程應用中經常有一些核心表需要按照多種形式查詢,如果為每一種查詢方式都建一個索引會影響表插入和更新的性能。

考慮到聯合索引在創建時每個子列都是排好序的,比如數據表A上有一個聯合縮影(a, b, c) , 那麼查詢where a = xxx ; where a = xxx and b = xxx都將命中縮影,因此可以利用這種特性按照業務需求設置少量聯合索引覆蓋多種查詢需求。

假設有表A, 有如下3種高頻查詢

select xx from A where a = xxx;
select xx from A where b = xxx;
select xx from A where a = xxx and b = xxx;

最簡單的辦法是 分別給a b (a, b) 建索引,但這就過於啰嗦。按照左首碼原則,最合理的索引建法應該是 b 和 (a, b)。

2.7 自增主鍵索引

1、InnoDB所有數據都是基於B+Tree存儲的,如果沒有主鍵mysql會在所有列中選擇可能唯一的列用作索 引id,如果查找不到會預設增加rowid列。

2、索引查找過程中會有大量數據比對的場景,如果使用uuid會逐位比對,效率會非常低,占用空間也會非 常大,占用過多ssd空間,存儲費用增大。

3、b+tree是有序樹,自增索引數據可以一直向後插入性能高,如果使用非自增索引,可能導致插入過程中 帶來樹分裂及平衡問題,帶來額外的性能損失。

3、常規資料庫優化順序

1、檢查SQL,查看執行計劃,是否命中索引?是否存在大量大表關聯?查詢的每個欄位都是必須的?...

2、加索引

3、分區

4、分表

5、改表結構,減少查詢種的關聯,增加冗餘欄位

6、加伺服器,彈性主機加U加記憶體換SSD...

 

點擊關註,第一時間瞭解華為雲新鮮技術~


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 目錄 一、條件判斷 二、邏輯判斷 三、if和case 四、七個實驗 一、條件判斷 1.test測試 test [ 條件表達式 ] -e:測試目錄是否存在 -d:測試是否為目錄 -f:是否為文件 -r:當前用戶是否有讀寫許可權 -w:當前用戶是否有寫許可權 -x:當前用戶是否有執行許可權 2.整數值判斷 格 ...
  • INFINI Labs 產品更新啦~,本次產品版本更新包括 Gateway v1.14.0、Console v1.2.0、Easysearch v1.1.1 等,其中 Console 在上一版基礎上做了很多優化改進以及新增了一些特性,如新增數據比對校驗功能、數據看板模塊新增了表格組件、圖表組件支持下 ...
  • 我們都知道,預設情況下,nginx的項目log是一直被累計寫入的,隨著時間越久,那麼這個文件就會越大,這個時候如果我們要去做一些查找和排查就會比較困難,因為日誌文件太大,操作起來比較費勁。 因此我們為了規避這個問題,提出日誌切割的方案。 那日誌切割的原理是怎麼樣的,我們來分析一下,我們先統計下連續1 ...
  • 在上篇文章 [《深入理解 slab cache 記憶體分配全鏈路實現》](https://mp.weixin.qq.com/s?__biz=Mzg2MzU3Mjc3Ng==&mid=2247488152&idx=1&sn=7c65f8ee28e9cc14a86e9df92b6d2b93&chksm=c ...
  • ![](https://img2023.cnblogs.com/blog/3076680/202305/3076680-20230516151258933-1445766867.png) # 1. 基本信息 SQL進階教程 [日]MICK 人民郵電出版社,2017年11月出版,**1版** ## 1 ...
  • ## Doris 簡介 ### Doris 概述 Apache Doris 由百度大數據部研發 (之前叫百度 Palo,2018 年貢獻到 Apache 社區後,更名為 Doris), 在百度內部,有超過 200 個產品線在使用,部署機器超過 1000 台,單一業務最大可達到上百 TB。 Apach ...
  • 我們介紹一下SeaTunnel支持的第一個同步場景:離線批量同步。顧名思意,離線批量同步需要用戶定義好SeaTunnel JobConfig,選擇批處理模式,作業啟動後開始同步數據,當數據同步完成後作業完成退出。 ...
  • 雖然感覺生活小知識沒有多少人喜歡看,這主要是因為生活小知識的策略不對,比如你要讓別人一時間接受20條還不如一天推一條,讓別人20天記住20條來得好,也不會讓訪客感到厭煩。 一級分類包含:安全急救(1445)、奧運知識(206)、服飾裝扮(512)、家電電腦(555)、健康養生(4390)、經營理財( ...
一周排行
    -Advertisement-
    Play Games
  • .Net8.0 Blazor Hybird 桌面端 (WPF/Winform) 實測可以完整運行在 win7sp1/win10/win11. 如果用其他工具打包,還可以運行在mac/linux下, 傳送門BlazorHybrid 發佈為無依賴包方式 安裝 WebView2Runtime 1.57 M ...
  • 目錄前言PostgreSql安裝測試額外Nuget安裝Person.cs模擬運行Navicate連postgresql解決方案Garnet為什麼要選擇Garnet而不是RedisRedis不再開源Windows版的Redis是由微軟維護的Windows Redis版本老舊,後續可能不再更新Garne ...
  • C#TMS系統代碼-聯表報表學習 領導被裁了之後很快就有人上任了,幾乎是無縫銜接,很難讓我不想到這早就決定好了。我的職責沒有任何變化。感受下來這個系統封裝程度很高,我只要會調用方法就行。這個系統交付之後不會有太多問題,更多應該是做小需求,有大的開發任務應該也是第二期的事,嗯?怎麼感覺我變成運維了?而 ...
  • 我在隨筆《EAV模型(實體-屬性-值)的設計和低代碼的處理方案(1)》中介紹了一些基本的EAV模型設計知識和基於Winform場景下低代碼(或者說無代碼)的一些實現思路,在本篇隨筆中,我們來分析一下這種針對通用業務,且只需定義就能構建業務模塊存儲和界面的解決方案,其中的數據查詢處理的操作。 ...
  • 對某個遠程伺服器啟用和設置NTP服務(Windows系統) 打開註冊表 HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Services\W32Time\TimeProviders\NtpServer 將 Enabled 的值設置為 1,這將啟用NTP伺服器功 ...
  • title: Django信號與擴展:深入理解與實踐 date: 2024/5/15 22:40:52 updated: 2024/5/15 22:40:52 categories: 後端開發 tags: Django 信號 松耦合 觀察者 擴展 安全 性能 第一部分:Django信號基礎 Djan ...
  • 使用xadmin2遇到的問題&解決 環境配置: 使用的模塊版本: 關聯的包 Django 3.2.15 mysqlclient 2.2.4 xadmin 2.0.1 django-crispy-forms >= 1.6.0 django-import-export >= 0.5.1 django-r ...
  • 今天我打算整點兒不一樣的內容,通過之前學習的TransformerMap和LazyMap鏈,想搞點不一樣的,所以我關註了另外一條鏈DefaultedMap鏈,主要調用鏈為: 調用鏈詳細描述: ObjectInputStream.readObject() DefaultedMap.readObject ...
  • 後端應用級開發者該如何擁抱 AI GC?就是在這樣的一個大的浪潮下,我們的傳統的應用級開發者。我們該如何選擇職業或者是如何去快速轉型,跟上這樣的一個行業的一個浪潮? 0 AI金字塔模型 越往上它的整個難度就是職業機會也好,或者說是整個的這個運作也好,它的難度會越大,然後越往下機會就會越多,所以這是一 ...
  • @Autowired是Spring框架提供的註解,@Resource是Java EE 5規範提供的註解。 @Autowired預設按照類型自動裝配,而@Resource預設按照名稱自動裝配。 @Autowired支持@Qualifier註解來指定裝配哪一個具有相同類型的bean,而@Resourc... ...