成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

MySQL內(nèi)核技術之“pthead局部變量”

KnewOne / 3141人閱讀

摘要:使用了稱之為的一系列文件和結(jié)構來進行監(jiān)控。的使用就像一個全局變量,哪個線程都可以用,但是實際上對應了線程內(nèi)部的變量值,可以參見該例。規(guī)定,線程存儲變量必須要先初始化。

MySQL使用了稱之為psi/pfs的一系列文件和結(jié)構來進行performance監(jiān)控。Psi全稱為performance schema interface,pfs全稱為performance storage。

該機制使用pthead來進行操作,其首先定義了pthread的線程存儲變量(pfs.cc):

thread_local_key_t THR_PFS;
thread_local_key_t THR_PFS_VG;   // global_variables
thread_local_key_t THR_PFS_SV;   // session_variables
thread_local_key_t THR_PFS_VBT;  // variables_by_thread
thread_local_key_t THR_PFS_SG;   // global_status
thread_local_key_t THR_PFS_SS;   // session_status
thread_local_key_t THR_PFS_SBT;  // status_by_thread
thread_local_key_t THR_PFS_SBU;  // status_by_user
thread_local_key_t THR_PFS_SBH;  // status_by_host
thread_local_key_t THR_PFS_SBA;  // status_by_account

bool THR_PFS_initialized= false;

這里的thread_local_key_t實際上是pthread_key_t,即pthread線程存儲變量。pthread_key_t的使用就像一個全局變量,哪個線程都可以用,但是實際上對應了線程內(nèi)部的變量值,可以參見該例:http://www.jianshu.com/p/d52c...。pthread規(guī)定,線程存儲變量thread_local_key_t必須要先初始化。MySQL在pfs_server.cc中對這些變量統(tǒng)一初始化:

void pre_initialize_performance_schema()
{
  pfs_initialized= false;

  init_all_builtin_memory_class();

  PFS_table_stat::g_reset_template.reset();
  global_idle_stat.reset();
  global_table_io_stat.reset();
  global_table_lock_stat.reset();

  if (my_create_thread_local_key(&THR_PFS, destroy_pfs_thread))
    return;
  if (my_create_thread_local_key(&THR_PFS_VG, NULL))  // global_variables
    return;
  if (my_create_thread_local_key(&THR_PFS_SV, NULL))  // session_variables
    return;
  if (my_create_thread_local_key(&THR_PFS_VBT, NULL)) // variables_by_thread
    return;
  if (my_create_thread_local_key(&THR_PFS_SG, NULL))  // global_status
    return;
  if (my_create_thread_local_key(&THR_PFS_SS, NULL))  // session_status
    return;
  if (my_create_thread_local_key(&THR_PFS_SBT, NULL)) // status_by_thread
    return;
  if (my_create_thread_local_key(&THR_PFS_SBU, NULL)) // status_by_user
    return;
  if (my_create_thread_local_key(&THR_PFS_SBH, NULL)) // status_by_host
    return;
  if (my_create_thread_local_key(&THR_PFS_SBA, NULL)) // status_by_account
    return;

  THR_PFS_initialized= true;
}

注意,這個初始化只做一次,以后創(chuàng)建線程時直接使用即可。上的第一個變量THR_PFS就是我們要使用的。

如何使用

使用上的方式初始化,首先要set相應的value:

/**
  @brief Execute the JOIN generated by parallel

  @param join [in] JOIN structure
*/
void execute_join(parallel_execution_thread_arg* parallel_arg) {
    /*
     * Get the parameter:
     * 1. JOIN
     * 2. pfs
     */
    /// TODO: do we need to handle error?
    std::cout << "****************I am in worker thread*****************" << std::endl;

    /// Get join
    JOIN* join= parallel_arg->join;

    /// Get and Set pfs
    PSI_thread* pfs= parallel_arg->pfs;
    pfs_set_thread_v1(pfs);

    /// Delete
    delete parallel_arg;
    
    /// Set the new thread context
    my_thread_set_THR_THD(join->thd);

    /// Execute
    join->exec();
}

上面的函數(shù)是我在MySQL中新加入的代碼,其中使用pfs_set_thread_v1進行set操作,即把當前THR_PFS對應的值設置為pfs。

get操作。由于我們加入了boost線程庫,所以當啟動一個線程時需要把JOIN結(jié)構和pfs結(jié)構傳入。思路是首先通過THR_PFS獲得pfs線程句柄,作為參數(shù)傳入到新的線程中。再新線程執(zhí)行函數(shù)中,把pfs線程句柄set進去。具體在sql_select.cc中,我們加入了如下代碼:

/**
  Parallel execution.

  @details When a JOIN is parallel, its JOINs will execute parallelly.
  Put all JOINs into thread pool to execute.
*/
void JOIN::parallel_exec_joins() {
  for (uint i= 0; i < m_parallel_joins.size(); i ++) {
    /// Delete it in join->exec
    parallel_execution_thread_arg* parallel_arg= new parallel_execution_thread_arg();

    /// Set join
    JOIN* join= m_parallel_joins[i];
    parallel_arg->join= join;

    /// Set pfs
    PSI_thread* pfs= pfs_get_thread_v1();
    parallel_arg->pfs= pfs;

    /// Thread pool
    generic_thread_pool.SubmitTask(execute_join, (parallel_execution_thread_arg* &&)parallel_arg);
  }
}

可以看到,我們通過MySQL的pfs_get_thread_v1獲得pfs線程句柄傳入到新的線程。

上面的例子是針對pfs的線程。對于MySQL普通線程的例子在上面的execute_join也能找到。注意里面有一行code:

/// Set the new thread context
my_thread_set_THR_THD(join->thd);

這里就是把當前的thd設置到pthread中。所以我們看到,在MySQL中的很多地方都用到了這個東西,用法也已經(jīng)明確了。

文章版權歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/17619.html

相關文章

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<