PHP中使用 fsockopen curl 模拟异步处理
PHP它最大缺點就是無法實現多線程管理,其程序的執行都是從頭到尾,按照邏輯一路執行下來,不可能出現分支,這一點是限制php在主流程序語言中往更高級的語言發展的原因之一。
在PHP中我們有的時候其實希望在執行某項操作的時候,同時去執行另外一項操作,舉一個場景:在用戶搶票的時候,你并不希望用戶排隊去連接數據庫進行查詢、判斷、插入,完成之后再返回用戶結果。其實我們并不需要用戶等那么久的時間,用戶提交之后,直接告訴他已經搶票成功了就可以了,至于各種操作,交給后臺去處理就好。當然,這種情況我們現在都用消息列表來處理,把每一個用戶提交的請求存在一個消息列隊中,告訴用戶已經搞定了,用戶愉快的關掉頁面之后,實際上后臺還在一個一個從消息列隊中取出請求進行操作。我們這篇文章則是通過一種異類的手法,實現操作在后臺運行,無需用戶等待。今天正好有這個需求。后臺(A服務器)的操作日志所有記錄在elk系統,elk裝在另外一臺服務器(B)上,打算在B上搭建laravel作為接受日志請求的api
話不多說(方案不是太好,請輕噴哦)
首先,我們要創建一個請求入口:
<?php提交的數據(后臺操作行為)提交給后臺API(B服務器)告訴管理員用戶已經搞定了其次,我們需要一個后臺api處理程序(B服務器),用戶是否在線并不影響它的運行: <?php ignore_user_abort(true); set_time_limit(0);過來的數據 數據處理
現在的問題是,在第一段代碼中,如何“提交給后臺”?我們通過一種非阻塞式的請求來實現這個功能。也就是創建一個可以被訪問的url,在這個url運行第二段程序,通過一個請求來請求這個url,從而激活第二段程序自動運行。接下來我們直接看代碼
比如我的操作后臺(A服務器):
// 遠程請求(不獲取內容)函數 function _sock($url,$post_data) {$host = parse_url($url,PHP_URL_HOST);$port = parse_url($url,PHP_URL_PORT);$port = $port ? $port : 80;$scheme = parse_url($url,PHP_URL_SCHEME);$path = parse_url($url,PHP_URL_PATH);$query = parse_url($url,PHP_URL_QUERY);if($query) $path .= '?'.$query;if($scheme == 'https') {$host = 'ssl://'.$host;}$method = "GET";if(!empty($post_data)){$method = "POST";}$fp = fsockopen($host,$port,$error_code,$error_msg,1);if(!$fp) {return array('error_code' => $error_code,'error_msg' => $error_msg);}else {stream_set_blocking($fp,false);//如果 mode 為0,資源流將會被轉換為非阻塞模式;如果是1,資源流將會被轉換為阻塞模式stream_set_timeout($fp,1);//設置超時$header = "$method $path HTTP/1.1\r\n";$header.="Host: $host\r\n";foreach($post_data as $k => $v){$_post[]= $k."=".urlencode($v);//必須做url轉碼以防模擬post提交的數據中有&符而導致post參數鍵值對紊亂 }$_post = implode('&', $_post);$header .= "Content-Type: application/x-www-form-urlencoded\r\n";//POST數據$header .= "Content-Length: ". strlen($_post) ."\r\n";//POST數據的長度$header.="Connection: close\r\n\r\n";//長連接關閉$header .= $_post; //傳遞POST數據fwrite($fp, $header);usleep(1000); // 這一句也是關鍵,如果沒有這延時,可能在nginx服務器上就無法執行成功,網上很多沒有帶這個參數。反正我幾次試過沒有設置這個是執行不了(連接主動斷開時,線上proxy層沒有及時把請求發給上游)fclose($fp);return array('error_code' => 0);} } $data = array('authId'=>'authcode','email'=>'zpt@php.net','nickname'=>'周伯通','mailBody'=>'<h3>通哥 <span style="padding:15px">某某的文章</span> 有新的留言</span></h3><p>周某人 < zpt@php.net >在評論中說:</p>' ); echo microtime(),"\r\n"; _sock('http://dev.yiqi.com/Activity/checkdata/cli_fix?m=Comment&a=sendEmail',$data);//dev.yiqihao.com比如是處理請求的api域名(B服務器),比如laravel構建 echo microtime();我們創建了一個基于fsockopen的函數,這個函數中利用fsockopen去訪問url,但是在訪問時,并不要求獲取url顯示的內容,而是僅僅發出訪問請求,請求到達后馬上關閉這個訪問。這樣做的好處就是無需再等待被訪問的url是否返回了可靠的信息,節約了時間,
這段代碼的執行時間在0.1-0.2秒之間,對于普通訪客而言,幾乎察覺不到。因此,在使用時,僅需要調用這個函數和對應的url即可。不過,這里并沒有提供數據傳輸的部分,如何傳輸數據,其實只需要在$header中增加post的內容即可。
接著上面,比如我的B服務器api :http://dev.yiqi.com/Activity/checkdata/cli_fix 處理簡單的邏輯:
<php echo file_put_contents("test.txt",$_REQUEST, FILE_APPEND);執行_sock后:
除了fsockopen,curl其實也可以實現這樣的效果,有些主機上并不支持fsockopen,我們就可以使用curl來實現。
function _curl($url) {$ch = curl_init();curl_setopt($ch,CURLOPT_URL,$url);curl_setopt($ch,CURLOPT_RETURNTRANSFER,1);curl_setopt($ch,CURLOPT_TIMEOUT,1);$result = curl_exec($ch);curl_close($ch);return $result; }這段代碼的關鍵是提供了一個Timeout,僅1秒鐘,也就是說curl發出請求,無論是否接收到返回的內容,1秒鐘之后都會關閉該訪問,因此這個函數的執行數據為1.0-1.1秒之間。但對于用戶來說,如果是一個需要進行數據處理的應用,1秒中的等待幾乎是被忽略的,如果你希望用一段更簡單和容易被理解的代碼,可以選擇curl來實現。這里就不給出示例了。有興趣的朋友可以試著寫一個哦.
總結
以上是生活随笔為你收集整理的PHP中使用 fsockopen curl 模拟异步处理的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: mac 命令行自动安装软件
- 下一篇: APP天气预报图标素材下载