中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

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

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

PHP如何實現并發查詢MySQL

發布時間:2021-07-20 14:39:24 來源:億速云 閱讀:417 作者:小新 欄目:開發技術

這篇文章主要為大家展示了“PHP如何實現并發查詢MySQL”,內容簡而易懂,條理清晰,希望能夠幫助大家解決疑惑,下面讓小編帶領大家一起研究并學習一下“PHP如何實現并發查詢MySQL”這篇文章吧。

同步查詢

這是我們最常的調用模式,客戶端調用Query[函數],發起查詢命令,等待結果返回,讀取結果;再發送第二條查詢命令,等待結果返回,讀取結果。總耗時,會是兩次查詢的時間之和。簡化一下過程,例如下圖:

PHP如何實現并發查詢MySQL 

例圖,由1.1到1.3為一個Query[函數]的調用,兩次查詢,就要串行經歷1.1、1.2、1.3、2.1、2.2、2.3,尤其在1.2和2.2會阻塞等待,進程沒法做其他事情。

同步調用的好處是,符合我們的直觀思維,調用和處理都簡單。缺點是進程阻塞在等待結果返回,增加額外的運行時間。
如果,有多條查詢請求,或者進程還有其他的事情處理,那么能否把等待的時間也合理利用起來,提高進程的處理能力呢,顯然是可以的。

拆分

現在,我們把Query[函數]打碎,客戶端在1.1后,馬上返回,客戶端跳過1.2,在1.3有數據達到后再去讀取數據。這樣進程在原來的1.2階段就解放了,可以做更多的事情,例如…再發起一條sql查詢[2.1],是否看到了并發查詢的雛形了。

并發查詢

相對于同步查詢的下一條查詢的發起都在上一條完成后,并發查詢,可以在上一條查詢請求發起后,立刻發起下一條查詢請求。簡化一下過程,下圖:

PHP如何實現并發查詢MySQL 

例圖,在1.1.1成功發送完請求后,立馬返回[1.1.2],最終查詢結果的返回時在遙遠的1.2 。但是在,1.1.1到1.2中間,還發起了另一個查詢請求,這時間段內,就同時發起了兩條查詢請求,2.2先于1.2到達,那么兩條查詢的總耗時,只相當于第一條查詢的時間。

并發查詢的優點是,可以提高進程的使用率,避免阻塞等待服務器處理查詢,縮短了多條查詢的耗時。但缺點也很明顯,發起N條并發查詢,就需要建立N條數據庫鏈接,對于有數據庫連接池的應用來說,可以避免這種情況。

退化

理想情況下,我們希望并發N條查詢,總耗時等于查詢時間最長的一條查詢。但也有可能并發查詢會[退化]為[同步查詢]。What?例圖中,如果1.2在2.1.1前就返回了,那么并發查詢就[退化]為[同步查詢]了,但付出的代價卻比同步查詢要高。

多路復用

  • 發起query1

  • 發起query2

  • 發起query3

  • ………

  • 等待query1、query2、query3

  • 讀取query2結果

  • 讀取query1結果

  • 讀取query3結果

那么,怎么等待知道什么時候查詢結果返回了,又是哪個的查詢結果返回呢?

對每個查詢IO調用read?如果是遇上阻塞IO,這樣就會阻塞在一個IO上,其他IO有結果返回了,也沒法處理。那么,如果是非阻塞IO,那不用怕會阻塞在其中一個IO上了,確實是,但又會造成不斷地輪詢判斷,浪費CPU資源。

對于這種情況可以使用多路復用輪詢多個IO。

PHP實現并發查詢MySQL

PHP的mysqli(mysqlnd驅動)提供多路復用輪詢IO(mysqli_poll)和異步查詢(MYSQLI_ASYNC、mysqli_reap_async_query),使用這兩個特性實現并發查詢,示例代碼:

<?php
 $sqls = array(
  'SELECT * FROM `mz_table_1` LIMIT 1000,10',
  'SELECT * FROM `mz_table_1` LIMIT 1010,10',
  'SELECT * FROM `mz_table_1` LIMIT 1020,10',
  'SELECT * FROM `mz_table_1` LIMIT 10000,10',
  'SELECT * FROM `mz_table_2` LIMIT 1',
  'SELECT * FROM `mz_table_2` LIMIT 5,1'
 );
 $links = [];
 $tvs = microtime();
 $tv = explode(' ', $tvs);
 $start = $tv[1] * 1000 + (int)($tv[0] * 1000);
 // 鏈接數據庫,并發起異步查詢
 foreach ($sqls as $sql) { 
  $link = mysqli_connect('127.0.0.1', 'root', 'root', 'dbname', '3306');
  $link->query($sql, MYSQLI_ASYNC); // 發起異步查詢,立即返回
  $links[$link->thread_id] = $link;
 }
 $llen = count($links);
 $process = 0;
 do {
  $r_array = $e_array = $reject = $links;
  // 多路復用輪詢IO
  if(!($ret = mysqli_poll($r_array, $e_array, $reject, 2))) {
   continue;
  }
  // 讀取有結果返回的查詢,處理結果
  foreach ($r_array as $link) {
   if ($result = $link->reap_async_query()) {
    print_r($result->fetch_row());
    if (is_object($result))
     mysqli_free_result($result);
   } else {
   }
   // 操作完后,把當前數據鏈接從待輪詢集合中刪除
   unset($links[$link->thread_id]);
   $link->close();
   $process++;
  }
  foreach ($e_array as $link) {
   die;
  }
  foreach ($reject as $link) {
   die;
  }
 }while($process < $llen);
 $tvs = microtime();
 $tv = explode(' ', $tvs);
 $end = $tv[1] * 1000 + (int)($tv[0] * 1000);
 echo $end - $start,PHP_EOL;

mysqli_poll源碼:

#ifndef PHP_WIN32
#define php_select(m, r, w, e, t) select(m, r, w, e, t)
#else
#include "win32/select.h"
#endif
/* {{{ mysqlnd_poll */
PHPAPI enum_func_status
mysqlnd_poll(MYSQLND **r_array, MYSQLND **e_array, MYSQLND ***dont_poll, long sec, long usec, int * desc_num)
{
 struct timeval tv;
 struct timeval *tv_p = NULL;
 fd_set   rfds, wfds, efds;
 php_socket_t max_fd = 0;
 int    retval, sets = 0;
 int    set_count, max_set_count = 0;
 DBG_ENTER("_mysqlnd_poll");
 if (sec < 0 || usec < 0) {
  php_error_docref(NULL, E_WARNING, "Negative values passed for sec and/or usec");
  DBG_RETURN(FAIL);
 }
 FD_ZERO(&rfds);
 FD_ZERO(&wfds);
 FD_ZERO(&efds);
 // 從所有mysqli鏈接中獲取socket鏈接描述符
 if (r_array != NULL) {
  *dont_poll = mysqlnd_stream_array_check_for_readiness(r_array);
  set_count = mysqlnd_stream_array_to_fd_set(r_array, &rfds, &max_fd);
  if (set_count > max_set_count) {
   max_set_count = set_count;
  }
  sets += set_count;
 }
 // 從所有mysqli鏈接中獲取socket鏈接描述符
 if (e_array != NULL) {
  set_count = mysqlnd_stream_array_to_fd_set(e_array, &efds, &max_fd);
  if (set_count > max_set_count) {
   max_set_count = set_count;
  }
  sets += set_count;
 }
 if (!sets) {
  php_error_docref(NULL, E_WARNING, *dont_poll ? "All arrays passed are clear":"No stream arrays were passed");
  DBG_ERR_FMT(*dont_poll ? "All arrays passed are clear":"No stream arrays were passed");
  DBG_RETURN(FAIL);
 }
 PHP_SAFE_MAX_FD(max_fd, max_set_count);
 // select輪詢阻塞時間
 if (usec > 999999) {
  tv.tv_sec = sec + (usec / 1000000);
  tv.tv_usec = usec % 1000000;
 } else {
  tv.tv_sec = sec;
  tv.tv_usec = usec;
 }
 tv_p = &tv;
 // 輪詢,等待多個IO可讀,php_select是select的宏定義
 retval = php_select(max_fd + 1, &rfds, &wfds, &efds, tv_p);
 if (retval == -1) {
  php_error_docref(NULL, E_WARNING, "unable to select [%d]: %s (max_fd=%d)",
      errno, strerror(errno), max_fd);
  DBG_RETURN(FAIL);
 }
 if (r_array != NULL) {
  mysqlnd_stream_array_from_fd_set(r_array, &rfds);
 }
 if (e_array != NULL) {
  mysqlnd_stream_array_from_fd_set(e_array, &efds);
 }
 // 返回可操作的IO數量
 *desc_num = retval;
 DBG_RETURN(PASS);
}

并發查詢操作結果

為了更直觀地看效果,我找了一個1.3億數據量并且沒有優化過的表進行操作。

并發查詢的結果:

PHP如何實現并發查詢MySQL 

同步查詢的結果:

PHP如何實現并發查詢MySQL 

從結果來看,同步查詢的總耗時是所有查詢的時間的累加;而并發查詢的總耗時在這里其實是查詢時間最長的那一條(同步查詢的第四條,耗時是10幾秒,符合并發查詢的總耗時),而且并發查詢的查詢順序和結果到達的順序是不一樣的。

多條耗時較短的查詢對比

使用多條查詢時間較短的sql進行對比一下

并發查詢的測試1結果(數據庫鏈接時間也統計進去):

PHP如何實現并發查詢MySQL 

同步查詢的結果(數據庫鏈接時間也統計進去):

PHP如何實現并發查詢MySQL 

并發查詢的測試2結果(不統計數據庫鏈接時間):

PHP如何實現并發查詢MySQL 

從結果上看,并發查詢測試1并沒有討到好處。從同步查詢上看,每條查詢耗時大概3-4ms左右。但如果不把數據庫鏈接時間統計進去(同步查詢只有一次數據庫鏈接),并發查詢的優勢又能體現出來了。

以上是“PHP如何實現并發查詢MySQL”這篇文章的所有內容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內容對大家有所幫助,如果還想學習更多知識,歡迎關注億速云行業資訊頻道!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

旬邑县| 扶风县| 赤峰市| 临汾市| 商水县| 西丰县| 阿拉善盟| 石家庄市| 宁陕县| 册亨县| 北宁市| 兰州市| 台湾省| 江门市| 镇沅| 关岭| 双辽市| 沛县| 平顶山市| 绵竹市| 临潭县| 富平县| 汶川县| 冕宁县| 济阳县| 龙门县| 海安县| 西藏| 都江堰市| 南岸区| 辉县市| 定襄县| 华坪县| 榆树市| 景宁| 古丈县| 绥芬河市| 湾仔区| 遂川县| 元阳县| 西藏|