溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

如何理解在MTS中Worker線程看到Time為負數(shù)

發(fā)布時間:2021-11-03 15:07:01 來源:億速云 閱讀:136 作者:柒染 欄目:MySQL數(shù)據(jù)庫

如何理解在MTS中Worker線程看到Time為負數(shù),針對這個問題,這篇文章詳細介紹了相對應(yīng)的分析和解答,希望可以幫助更多想解決這個問題的小伙伴找到更簡單易行的方法。

一、問題來源

這是一個朋友問我的一個問題,問題如下,在MTS中Worker線程看到Time為負數(shù)是怎么回事,如下:

如何理解在MTS中Worker線程看到Time為負數(shù)

二、關(guān)于show processlist中的Time

實際上show processlist中的信息基本都來自函數(shù) mysqld_list_processes,也就是說每次執(zhí)行show processlist 都需要執(zhí)行這個函數(shù)來進行填充。對于Time值來講它來自如下信息:

Percona :
time_t now= my_time(0);
 protocol->store_long ((thd_info->start_time > now) ? 0
        : (longlong) (now - thd_info->start_time));
官方版:
time_t now= my_time(0);
protocol->store_long ((longlong) (now - thd_info->start_time));

我們可以注意到在Percona的版本中對這個輸出值做了優(yōu)化,也就是如果出現(xiàn)負數(shù)的時候直接顯示為0,但是官方版中沒有這樣做,可能出現(xiàn)負數(shù)。

三、計算方式解讀和測試

現(xiàn)在我們來看看這個簡單的計算公式,實際上now很好理解就是服務(wù)器的當(dāng)前時間,重點就在于thd_info->start_time的取值來自何處。
實際這個時間來自于函數(shù)THD::set_time,但是需要注意的是這個函數(shù)會進行重載,有下面三種方式:

重載1

inline void set_time()
  {
    start_utime= utime_after_lock= my_micro_time();
    if (user_time.tv_sec || user_time.tv_usec)
    {
      start_time= user_time;
    }
    else
      my_micro_time_to_timeval(start_utime, &start_time);
...
  }

重載2

  inline void set_time(const struct timeval *t)
  {
    start_time= user_time= *t;
    start_utime= utime_after_lock= my_micro_time();
...
  }

重載3

  void set_time(QUERY_START_TIME_INFO *time_info)
  {
    start_time= time_info->start_time;
    start_utime= time_info->start_utime;
  }

其實簡單的說就是其中有一個start_utime,如果設(shè)置了start_utime那么start_time將會指定為start_utime,并且在重載1中將會不會修改start_time,這一點比較重要。

好了說了3種方式,我們來看看Time的計算有如下可能。

1、執(zhí)行命令

如果主庫執(zhí)行常見的命令都會在命令發(fā)起的時候調(diào)用重載1,設(shè)置start_time為命令開始執(zhí)行的時間如下:

堆棧:
#0  THD::set_time (this=0x7ffe7c000c90) at /mysqldata/percona-server-locks-detail-5.7.22/sql/sql_class.h:3505
#1  0x00000000015c5fe8 in dispatch_command (thd=0x7ffe7c000c90, com_data=0x7fffec03fd70, command=COM_QUERY)
    at /mysqldata/percona-server-locks-detail-5.7.22/sql/sql_parse.cc:1247

可以看到這個函數(shù)沒有實參,因此start_time會設(shè)置為當(dāng)前時間,那Time的計算公式 now - thd_info->start_time就等于 (服務(wù)器當(dāng)前時間 - 命令開始執(zhí)行的時間)。

2、從庫 單Sql線程 和 Worker線程

其實不管單Sql線程還是Worker線程都是執(zhí)行Event的,這里的start_time將會被設(shè)置為Event header中timestamp的時間(query event/dml event),這個時間實際就是主庫命令發(fā)起的時間。如下:

堆棧:
query event:
#0  THD::set_time (this=0x7ffe78000950, t=0x7ffe701ec430) at /root/mysqlall/percona-server-locks-detail-5.7.22/sql/sql_class.h:3526
#1  0x00000000018459ab in Query_log_event::do_apply_event (this=0x7ffe701ec310, rli=0x7ffe7003c050, query_arg=0x7ffe701d88a9 "BEGIN", q_len_arg=5)
    at /root/mysqlall/percona-server-locks-detail-5.7.22/sql/log_event.cc:4714
堆棧:
dml event:
#0  THD::set_time (this=0x7ffe78000950, t=0x7ffe701ed5b8) at /root/mysqlall/percona-server-locks-detail-5.7.22/sql/sql_class.h:3526
#1  0x000000000185aa6e in Rows_log_event::do_apply_event (this=0x7ffe701ed330, rli=0x7ffe7003c050)
    at /root/mysqlall/percona-server-locks-detail-5.7.22/sql/log_event.cc:11417

我們看到這里有一個實參的傳入我們看一下代碼如下:

thd->set_time(&(common_header->when))

實際上就是這一行,這是我們前面說的重載3,這樣設(shè)置后start_utime和start_time都將會設(shè)置,即便調(diào)用重載1也不會更改, 那Time的計算方式 now - thd_info->start_time就等于 (從庫服務(wù)器當(dāng)前時間 - Event header中的時間),但是要知道 Event header中的時間實際也是來自于主庫命令發(fā)起的時間。 既然如此如果從庫服務(wù)器的時間小于主庫服務(wù)器的時間,那么Time的結(jié)果可能是負數(shù)是可能出現(xiàn)的,當(dāng)然Percona版本做了優(yōu)化負數(shù)將會顯示為0,如果從庫服務(wù)器的時間大于主庫的時間那么可能看到Time比較大。

因為我的測試環(huán)境都是Percona,為了效果明顯,我們來測試一下Worker線程Time很大的情況,如下設(shè)置:

主庫:
[root@mysqltest2 test]# date
Fri Nov  1 01:40:54 CST 2019
從庫:
[root@gp1 log]# date 
Tue Nov 19 15:58:37 CST 2019

主庫隨便做一個命令,然后觀察如下:

如何理解在MTS中Worker線程看到Time為負數(shù)

3、設(shè)置timestamp

如果手動指定timestamp也會影響到Time的計算結(jié)果,因為start_utime和start_time都將會設(shè)置,如下:

mysql> set timestamp=1572540000
堆棧:
#0  THD::set_time (this=0x7ffe7c000c90, t=0x7fffec03db30) at /mysqldata/percona-server-locks-detail-5.7.22/sql/sql_class.h:3526
#1  0x000000000169e509 in update_timestamp (thd=0x7ffe7c000c90, var=0x7ffe7c006860) at /mysqldata/percona-server-locks-detail-5.7.22/sql/sys_vars.cc:4966
#2  0x00000000016b9a3d in Sys_var_session_special_double::session_update (this=0x2e68e20, thd=0x7ffe7c000c90, var=0x7ffe7c006860)
    at /mysqldata/percona-server-locks-detail-5.7.22/sql/sys_vars.h:1889

我們看到帶入了實參,我們看看代碼這一行如下:

 thd->set_time(&tmp);

這就是重載2了,這樣設(shè)置后start_utime和start_time都將會設(shè)置,即便調(diào)用重載1也不會更改,言外之意就是設(shè)置了timestamp后即便執(zhí)行了其他的命令Time也不會更新。Time的計算方式 now - thd_info->start_time就等于 (服務(wù)器當(dāng)前時間 - 設(shè)置的timestamp時間),這樣的話就可能出現(xiàn)Time出現(xiàn)異常,比如很大或者為負數(shù)(Percona為0)如下:

如何理解在MTS中Worker線程看到Time為負數(shù)

4、空閑情況下

如果我們的會話空閑狀態(tài)下那么now - thd_info->start_time 公式中,now會不斷變大,但是thd_info->start_time卻不會改變,因此Time 會不斷增大,等待到下一次命令到來后才會更改。

四、延伸

這里我想在說明一下如果從庫開啟了log_slave_updates 的情況下,從庫記錄會記錄來自主庫的Event,但是這些Event的timestamp和Query Event的exetime如何取值呢?

Event的timestamp的取值

其實上面我已經(jīng)說了,因為 start_time將會被設(shè)置為Event header中timestamp的時間(query event/dml event),當(dāng)記錄Evnet的時候這個時間和主庫基本一致,如下:

如何理解在MTS中Worker線程看到Time為負數(shù)

很明顯我們會發(fā)現(xiàn)這些Event的timestamp不是本地的時間,而是主庫的時間。

Query Event的exetime

我們先來看看這個時間的計算方式:

  ulonglong micro_end_time= my_micro_time();//這里獲取時間 query event
  my_micro_time_to_timeval(micro_end_time, &end_time);
  exec_time= end_time.tv_sec - thd_arg->start_time.tv_sec;//這里計算時間

相信對于  thd_arg->start_time而言已經(jīng)不再陌生,它就是主庫命令發(fā)起的時間。我在我的《深入理解主從原理》系列中說過了,對于Query Event的exetime在 row格式binlog下,DML語句將會是第一行語句修改時間的時間,那么我們做如下定義(row格式 DML語句):

  • 主:主庫第一行數(shù)據(jù)修改完成的服務(wù)器時間 - 主庫本命令發(fā)起的時間

  • 從:從庫第一行數(shù)據(jù)修改完成的服務(wù)器時間 - 主庫本命令發(fā)起的時間

他們的差值就是:
(從庫第一行數(shù)據(jù)修改完成的服務(wù)器時間 - 主庫第一行數(shù)據(jù)修改完成的服務(wù)器時間 )

同樣如果我們從庫的時間遠遠大于主庫的時間,那么exetime也會出現(xiàn)異常如下:

如何理解在MTS中Worker線程看到Time為負數(shù)

Time是我們平時關(guān)注的一個指標(biāo),我們經(jīng)常用它來表示我的語句執(zhí)行了多久,但是如果出現(xiàn)異常的情況我們也應(yīng)該能夠明白為什么,這里我將它的計算方式做了一個不完全的解釋。

關(guān)于如何理解在MTS中Worker線程看到Time為負數(shù)問題的解答就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,如果你還有很多疑惑沒有解開,可以關(guān)注億速云行業(yè)資訊頻道了解更多相關(guān)知識。

向AI問一下細節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI