国产 无码 综合区,色欲AV无码国产永久播放,无码天堂亚洲国产AV,国产日韩欧美女同一区二区

如何進行MySQL源碼調(diào)試(一條select語句的執(zhí)行流程)

這篇具有很好參考價值的文章主要介紹了如何進行MySQL源碼調(diào)試(一條select語句的執(zhí)行流程)。希望對大家有所幫助。如果存在錯誤或未考慮完全的地方,請大家不吝賜教,您也可以點擊"舉報違法"按鈕提交疑問。

一、背景

MySQL是當(dāng)今世上最受歡迎的使用最廣泛的開源數(shù)據(jù)庫,它的繁榮離不開它的開源特性。放在過去商業(yè)數(shù)據(jù)庫的時代,大家都沒有機會接觸到數(shù)據(jù)庫的源代碼,但在如今開源數(shù)據(jù)庫的時代,越來越多的人開始研究數(shù)據(jù)庫的源碼,并給社區(qū)貢獻代碼,MySQL官方每次發(fā)布新版本都要感謝一些在社區(qū)上貢獻代碼的程序員?,F(xiàn)在新的數(shù)據(jù)庫時代也給DBA提出了更高的要求,學(xué)會調(diào)試源碼,通過源碼定位問題,這是DBA進階的方向。MySQL的源碼有幾百上千萬行,想全部搞懂幾乎是不可能的,研究源碼一般推薦從某個功能點入手。而學(xué)會調(diào)試源碼,不管對研究源碼或通過源碼定位問題,都是必備的技能。本文將介紹Linux平臺下如何通過gdb進行MySQL源碼調(diào)試,并簡單介紹通過調(diào)試源碼定位一條select語句的執(zhí)行流程。

二、源碼調(diào)試方法

關(guān)于源碼的編譯及調(diào)試,不同的平臺可以通過不同的工具來進行:windows平臺下可以通過visual studio來進行,https://www.cnblogs.com/huangxincheng/p/13084736.html 按照這篇文章的方法可以進行5.7的調(diào)試,通過cmake生成mysql.sln文件,然后給打開sln文件編譯成功后,就可以進入調(diào)試模式了。在某個函數(shù)處打下斷點,F(xiàn)5進入調(diào)試模式后,就可以對數(shù)據(jù)庫進行操作了,命中斷點后就可以查看堆棧的函數(shù)調(diào)用關(guān)系。一般在實際應(yīng)用中,MySQL都是運行在Linux平臺下,在Linux平臺下一般是通過GDB(GNU symbolic debugger)工具進行調(diào)試,C/C++ 項目的開發(fā)和調(diào)試包括故障排查都是利用 GDB 完成的。此外,VsCODE這種IDE工具可以在本地的windows操作系統(tǒng)下,通過ssh遠程調(diào)試Linux平臺下的MySQL,https://mp.weixin.qq.com/s/RO_Ipa9_SH8_DuVholrgvg,姜老師這篇文章也簡單講了一下如何操作。

三、GDB調(diào)試MySQL源碼

3.1 MySQL源碼下載

首先需要在MySQL官網(wǎng)上下載MySQL源碼,操作系統(tǒng)選擇為source code,OS版本選擇為ALL OPERATING SYSTEM,下載帶boost頭文件的源碼包。如果對MySQL的版本沒有特別要求的話,一般推薦下載最新版本的,因為老版本中存在bug的概率較大,編譯過程需要解決這些bug,比如在8.0.23版本中編譯過程中報了這個錯:buf0buf.cc:1227:44: error: ‘SYS_gettid’ was not declared in this scope。參考MySQL官方論壇:https://forums.mysql.com/read.php?117,674410,676378#msg-676378,在storage/innobase/buf/buf0flu.cc文件代碼中加上聲明#include <sys/syscall.h>,解決了這個報錯。

3.2 編譯安裝環(huán)境準(zhǔn)備

源碼編譯需要gcc等基礎(chǔ)軟件支持,MySQL8.0版本 gcc至少需要7.1以上,一般推薦直接升級到最高版本,在線環(huán)境比較好辦直接通過yum安裝即可。離線環(huán)境的下載及安裝可以參考這兩篇文章分別升級gcc和cmake:

https://mp.weixin.qq.com/s/kWPD1o5fWTAxUNETGsXA7ghttps://mp.weixin.qq.com/s/5sHPJ0Kxkmj2llW-UvEbkg。把gcc和cmake搞定后,就可以開始編譯安裝mysql了。

3.3 編譯和安裝MySQL

參考官方文檔通過源碼安裝mysql這一章可以完成mysql的編譯和安裝:

如何進行MySQL源碼調(diào)試(一條select語句的執(zhí)行流程)

實際cmake加上這幾個參數(shù),其中-DWITH_DEBUG=1是為了開啟調(diào)試模式。

cmake .. -DCMAKE_INSTALL_PREFIX=/usr/local/mysql -DWITH_BOOST=/root/gdb_mysql/mysql-8.0.23/boost/boost_1_73_0 -DWITH_DEBUG=1

接著make install成功后,配置好配置文件/etc/my.cnf,就可以初始化數(shù)據(jù)庫并啟動數(shù)據(jù)庫了。

mysqld --initialize --user=mysql
mysqld_safe --user=mysql &

啟動完數(shù)據(jù)庫后,登錄數(shù)據(jù)庫可以發(fā)現(xiàn)現(xiàn)在已經(jīng)是debug模式了。

如何進行MySQL源碼調(diào)試(一條select語句的執(zhí)行流程)

3.4 gdb調(diào)試源碼

完成MySQLdebug版本的安裝和啟動后,gdb命令下attach mysql的進程號,就可以對mysql進程進行打斷點調(diào)試了。

如何進行MySQL源碼調(diào)試(一條select語句的執(zhí)行流程)

?gdb調(diào)試過程中常用的命令可以參考如下:

attach 進程號 #進入調(diào)試模式
b Sql_cmd_insert::mysql_insert  #在某個函數(shù)打下斷點
b filename:linenum  #在文件的某行打下斷點    
clear function  #在某個函數(shù)處刪除斷點
bt   #查看堆棧信息
n    #next 單步調(diào)試,每次只執(zhí)行往下一行代碼,對于調(diào)用的函數(shù)來說,next 命令只會將其視作一行代碼。 #n 3 往下執(zhí)行三行代碼 
s    #step 單步調(diào)試,當(dāng) step 命令所執(zhí)行的代碼行中包含函數(shù)時,會進入該函數(shù)內(nèi)部,并在函數(shù)第一行代碼處停止執(zhí)行。
c    #continue 當(dāng)程序在某一斷點處停止運行后,使用該指令可以繼續(xù)執(zhí)行,直至遇到下一個斷點或者程序結(jié)束。
l    #list 顯示源程序代碼的內(nèi)容,包括各行代碼所在的行號。
p xxx  #print 打印指定xxx變量的值
info breakpoint   #查看斷點信息

?

四、一條select語句的調(diào)試

在handle_connection函數(shù)處打下斷點,然后在mysql客戶端執(zhí)行一條select語句,可以在gdb里面看到,程序很快命中了斷點,接下來便可以通過單步調(diào)試,一步一步定位select語句的執(zhí)行流程。

#客戶端連接的線程處理函數(shù)
handle_connection (arg=0xb998240) at /gdb/mysql-8.0.23/sql/conn_handler/connection_handler_per_thread.cc
#讀取連接發(fā)來的命令,然后執(zhí)行
do_command (thd=0x7f5a3815d3c0) at /gdb/mysql-8.0.23/sql/sql_parse.cc:1320
#THD類,描述每個客戶端連接產(chǎn)生的后臺進程
#發(fā)出命令
dispatch_command (thd=0x7f5a3815d3c0, com_data=0x7f5afc7adb00, command=COM_QUERY)
#根據(jù)command=COM_QUERY,調(diào)用alloc_query函數(shù)(讀取查詢語句并存在thd->query中)
if (alloc_query(thd, com_data->com_query.query,com_data->com_query.length))
#執(zhí)行到dispatch_sql_command(thd, &parser_state); 解析sql語句,然后把結(jié)果發(fā)給executor
dispatch_sql_command(thd, &parser_state);
#進入dispatch_sql_command函數(shù),執(zhí)行到mysql_execute_command函數(shù)(Execute command saved in thd and lex->sql_command)
error = mysql_execute_command(thd, true);

#在mysql_execute_command函數(shù)中,執(zhí)行到case SQLCOM_SELECT,res = lex->m_sql_cmd->execute(thd),進入到execute函數(shù)
#在mysql_execute_command函數(shù)中,switch (lex->sql_command)通過case SQLCOM_XXX,轉(zhuǎn)到不同語句的執(zhí)行器
#這時候就進入到了lex的公共屬性m_sql_cmd類下面的execute函數(shù);
#通過單步調(diào)試,此時程序進入到了Sql_cmd_dml::execute (this=0x7f5a38bee0b0, thd=0x7f5a3815d3c0) at /gdb/mysql-8.0.23/sql/sql_select.cc:517

#此時可以看到,解析SQL是在dispatch_sql_command和mysql_execute_command函數(shù)中完成的,Sql_cmd_dml::execute的函數(shù)主要有6步
Prelocking;Preparation;Locking of tables;Optimization;Execution or explain;Cleanup
#lock_tables(thd, lex->query_tables, lex->table_count, 0) 鎖表
#execute_inner(thd) 執(zhí)行

#進入execute階段
Sql_cmd_dml::execute_inner (this=0x7f1ca0011858, thd=0x7f1ca0005ed0) at /gdb/mysql-8.0.23/sql/sql_select.cc:809
if (unit->optimize(thd, /*materialize_destination=*/nullptr, /*create_iterators=*/true)) #優(yōu)化
if (unit->execute(thd)) return true; #執(zhí)行
#此時執(zhí)行到了SELECT_LEX_UNIT::execute (this=0x7f1ca0023e48, thd=0x7f1ca0005ed0) at /gdb/mysql-8.0.23/sql/sql_union.cc:1267
return ExecuteIteratorQuery(thd);
#執(zhí)行ExecuteIteratorQuery這個函數(shù)
SELECT_LEX_UNIT::ExecuteIteratorQuery (this=0x7f1ca0023e48, thd=0x7f1ca0005ed0) at /gdb/mysql-8.0.23/sql/sql_union.cc:1125
#執(zhí)行完成后返回查詢語句的結(jié)果值 return query_result->send_eof(thd);
#這個時候Sql_cmd_dml::execute_inner函數(shù)也執(zhí)行完成了,進而Sql_cmd_dml::execute,mysql_execute_command也接著執(zhí)行完成

#執(zhí)行到dispatch_command函數(shù)的thd->send_statement_status(); 這一行,看到客戶端執(zhí)行的查詢語句也輸出了結(jié)果
#執(zhí)行到handle_connection的while (thd_connection_alive(thd)) {if (do_command(thd)) break;}; 意味著mysql連接結(jié)束了,這時候調(diào)試也隨之結(jié)束。

select語句的執(zhí)行流程可以總結(jié)如下,這些函數(shù)可以方便以后打斷點,更快的定位問題:

1. 客戶端連接線程處理函數(shù)
handle_connection (arg=0xb998240) at /gdb/mysql-8.0.23/sql/conn_handler/connection_handler_per_thread.cc:301
2. 讀取連接發(fā)來的命令,然后執(zhí)行
do_command (thd=0x7f5a3815d3c0) at /gdb/mysql-8.0.23/sql/sql_parse.cc:1320
3. 發(fā)出命令,并將查詢語句存在thd->query中
dispatch_command (thd=0x7f1ca0011100, com_data=0x7f1d644d3b00, command=COM_QUERY) at /gdb/mysql-8.0.23/sql/sql_parse.cc:1836
4. 解析sql語句,然后把結(jié)果發(fā)給executor
dispatch_sql_command (thd=0x7f1ca0011100, parser_state=0x7f1d644d2a60) at /gdb/mysql-8.0.23/sql/sql_parse.cc:4988
5. 執(zhí)行存在thd中的語句
mysql_execute_command (thd=0x7f1ca0011100, first_level=true) at /gdb/mysql-8.0.23/sql/sql_parse.cc:4407
6. SELECT語句的:準(zhǔn)備,鎖表,優(yōu)化,執(zhí)行
Sql_cmd_dml::execute (this=0x7f1ca09feb28, thd=0x7f1ca0011100) at /gdb/mysql-8.0.23/sql/sql_select.cc:612
7. SELECT語句的優(yōu)化和執(zhí)行
Sql_cmd_dml::execute_inner (this=0x7f1ca0011858, thd=0x7f1ca0005ed0) at /gdb/mysql-8.0.23/sql/sql_select.cc:809
8. SELECT語句的執(zhí)行
SELECT_LEX_UNIT::execute (this=0x7f1ca0023e48, thd=0x7f1ca0005ed0) at /gdb/mysql-8.0.23/sql/sql_union.cc:1267
9. 執(zhí)行語句,返回結(jié)果
SELECT_LEX_UNIT::ExecuteIteratorQuery (this=0x7f1ca0023e48, thd=0x7f1ca0005ed0) at /gdb/mysql-8.0.23/sql/sql_union.cc:1125

五、總結(jié)

不管是研究MySQL源碼還是通過源碼定位問題,學(xué)會調(diào)試MySQL源碼都是必備的基礎(chǔ)技能,MySQL源碼體系十分龐大,調(diào)試源碼可以更快更清晰從源碼中定位問題。文章來源地址http://www.zghlxwxcb.cn/news/detail-437976.html

到了這里,關(guān)于如何進行MySQL源碼調(diào)試(一條select語句的執(zhí)行流程)的文章就介紹完了。如果您還想了解更多內(nèi)容,請在右上角搜索TOY模板網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章,希望大家以后多多支持TOY模板網(wǎng)!

本文來自互聯(lián)網(wǎng)用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務(wù),不擁有所有權(quán),不承擔(dān)相關(guān)法律責(zé)任。如若轉(zhuǎn)載,請注明出處: 如若內(nèi)容造成侵權(quán)/違法違規(guī)/事實不符,請點擊違法舉報進行投訴反饋,一經(jīng)查實,立即刪除!

領(lǐng)支付寶紅包贊助服務(wù)器費用

相關(guān)文章

  • MYSQL實戰(zhàn)45講筆記--基礎(chǔ)架構(gòu):一條SQL查詢語句是如何執(zhí)行的?

    MYSQL實戰(zhàn)45講筆記--基礎(chǔ)架構(gòu):一條SQL查詢語句是如何執(zhí)行的?

    MySQL 可以分為 Server 層和存儲引擎層兩部分。 Server 層 :連接器、查詢緩存、分析器、優(yōu)化器、執(zhí)行器等,涵蓋 MySQL 的大多數(shù)核心服務(wù)功能,以及所有的內(nèi)置函數(shù)(如日期、時間、數(shù)學(xué)和加密函數(shù)等),所有跨存儲引擎的功能都在這一層實現(xiàn),比如存儲過程、觸發(fā)器、視圖等

    2024年02月07日
    瀏覽(23)
  • MySQL—一條查詢SQL語句的完整執(zhí)行流程

    MySQL—一條查詢SQL語句的完整執(zhí)行流程

    表結(jié)構(gòu)和數(shù)據(jù)如下: 我們分析的sql語句如下: 大體來說,MySQL可以分為Server層和存儲引擎層兩部分: Server層 包括:連接器、查詢緩存、分析器、優(yōu)化器、執(zhí)行器等 涵蓋MySQL的大多數(shù)核心服務(wù)功能 所有的內(nèi)置函數(shù)(如日期、時間、數(shù)學(xué)和加密函數(shù)等),所有跨存儲引擎的功能都在

    2024年04月28日
    瀏覽(23)
  • MySQL面試題:一條SQL語句在MySQL中執(zhí)行過程全解析

    MySQL面試題:一條SQL語句在MySQL中執(zhí)行過程全解析

    介紹一下下圖涉及的一些組件的基本作用幫助大家理解這幅圖。 連接/線程處理(連接器): 身份認證和權(quán)限相關(guān)(如連接處理、授權(quán)認證、安全等等)。 查詢緩存: 執(zhí)行查詢語句的時候,會先查詢緩存(MySQL 8.0 版本后移除)。 解析器: 沒有命中緩存的話,SQL 語句就會經(jīng)過解析

    2024年02月03日
    瀏覽(52)
  • MySQL Select 語句執(zhí)行順序

    一條 SQL 查詢語句結(jié)構(gòu)如下: 但真正的執(zhí)行步驟如下,執(zhí)行時,每個步驟都會產(chǎn)生一個虛擬表,該虛擬表被用作下一個步驟的輸入: 1. FROM 在這一步,無論后面跟那種聯(lián)接運算(LEFT JOIN、RIGHT JOIN等),都首先執(zhí)行交叉聯(lián)接(CROSS JOIN),計算笛卡爾積,生成虛擬表 VT-1 2. ON 根

    2024年02月05日
    瀏覽(31)
  • 01 | 一條 SQL 查詢語句是如何執(zhí)行的?

    01 | 一條 SQL 查詢語句是如何執(zhí)行的?

    以下內(nèi)容出自 《MySQL 實戰(zhàn) 45 講》 一條 SQL 查詢語句是如何執(zhí)行的? 下面是 MySQL 的基本架構(gòu)示意圖,從中可以清楚地看到 SQL 語句在 MySQL 的各個功能模塊中的執(zhí)行過程。 大體來說,MySQL 可以分為 Server 層和存儲引擎層兩部分。 Server 層包括連接器、查詢緩存、分析器、優(yōu)化器

    2024年02月10日
    瀏覽(24)
  • 【MySQL系列】- SELECT語句執(zhí)行順序

    【MySQL系列】- SELECT語句執(zhí)行順序

    2.1 執(zhí)行FROM操作 這一步需要做的是對FROM子句前后的兩張表進行笛卡爾積操作,也稱作為交叉連接,生成虛擬表VT1。如果FROM子句前的表包含a行數(shù)據(jù),F(xiàn)ROM子句后的表中包含b行數(shù)據(jù),那么虛擬表VT1將包含a*b行數(shù)據(jù)。 2.2 應(yīng)用ON過濾器 SELECT查詢共有3個過濾流程,分別是ON、WHERE、

    2024年02月08日
    瀏覽(44)
  • 01 | 基礎(chǔ)架構(gòu):一條SQL查詢語句是如何執(zhí)行的?

    畫出 MySQL 的基本架構(gòu)圖 Server 層和存儲引擎層各自有哪些組件? Server 層 連接器 查詢緩存 分析器 優(yōu)化器 執(zhí)行器 所有內(nèi)置函數(shù)(日期、時間、數(shù)學(xué)和加密函數(shù)等) 所有跨存儲引擎的功能(存儲過程、觸發(fā)器、視圖等) 存儲引擎層 負責(zé)數(shù)據(jù)的存儲和讀取 連接器的作用? 連接

    2024年02月08日
    瀏覽(23)
  • MySQL中,SQL 查詢總是先執(zhí)行SELECT語句嗎?

    在使用 MySQL 進行查詢時,我們通常會使用 SELECT 語句,但是 SELECT 語句是否總是最先被執(zhí)行呢?這是一個非常有趣的問題,本文將對此進行探討。 在 MySQL 中,SQL 查詢通常包括以下幾個步驟: 語法解析 :MySQL 會對 SQL 查詢語句進行語法解析,檢查語句是否符合 SQL 語法規(guī)范。

    2023年04月09日
    瀏覽(21)
  • 一條SQL在MySQL中是如何執(zhí)行的

    一條SQL在MySQL中是如何執(zhí)行的

    MySQL的內(nèi)部組件結(jié)構(gòu) 連接數(shù)據(jù)庫做了什么? 建立鏈接的時候需要將host,user,pwd,在user表中校驗用戶(用戶名和密碼)是否正確。每個鏈接鏈接到服務(wù)器端之后會給鏈接開辟一個專屬的空間。這個空間存儲了很多這個鏈接需要使用的操作。并且會把這個用戶的權(quán)限都放到這個空間

    2023年04月09日
    瀏覽(95)
  • 一條SQL如何被MySQL架構(gòu)中的各個組件操作執(zhí)行的?

    一條SQL如何被MySQL架構(gòu)中的各個組件操作執(zhí)行的?

    簡單用一張圖說明下, MySQL 架構(gòu)有哪些組件,接下來給大家用 SQL 語句分析 假如 SQL 語句是這樣 SELECT class_no FROM student WHERE name = \\\'lcy\\\' AND age 18 GROUP BY class_no 其中 name 為索引,我們按照 時間順序 來分析一下 客戶端:客戶端(如 MySQL 命令行工具、 Navicat 、 MySQL Workbench 或其他應(yīng)

    2023年04月22日
    瀏覽(25)

覺得文章有用就打賞一下文章作者

支付寶掃一掃打賞

博客贊助

微信掃一掃打賞

請作者喝杯咖啡吧~博客贊助

支付寶掃一掃領(lǐng)取紅包,優(yōu)惠每天領(lǐng)

二維碼1

領(lǐng)取紅包

二維碼2

領(lǐng)紅包