首頁 >後端開發 >php教程 >如何使用fscok實現非同步呼叫PHP的程式碼案例

如何使用fscok實現非同步呼叫PHP的程式碼案例

黄舟
黄舟原創
2017-03-30 10:03:321568瀏覽

Web 伺服器執行一個腳本,可能幾毫秒就完成,也可能幾分鐘都完不成。如果程式執行緩慢,使用者可能沒有耐心等下去,就關閉瀏覽器了。

    而有的時候,我們更本不關心這些耗時的腳本的執行結果,但卻還要等他執行完返回,才能繼續下一步。  
    那麼有沒有辦法,只是簡單的觸發呼叫這些耗時的腳本然後就繼續下一步,讓這些耗時的腳本在服務端慢慢執行?
    
    接下來,我將使用fscokopen來實現這項功能。
   
    PHP是支援socket程式設計的,就是fsockopen,以前做CMS的時候,也曾經用它做過smtp發信。
    fscokopen傳回一個到遠端主機連線的句柄。你可以像使用fopen傳回的句柄一樣,對她進行寫fwrite,讀取fgets, fread等操作。
   
    我們的非同步PHP,而主要想要的效果是,觸發一個PHP腳本,然後立即返回,留它在伺服器端慢慢執行。前面我也寫過一篇文章討論過這個問題。

    那麼,我們就可以使用fsockopen連接到本機伺服器,觸發腳本執行,然後立即返回,不等待腳本執行完成。

function triggerRequest($url, $post_data = array(), $cookie = array())…{
        $method = "GET";  //可以通过POST或者GET传递一些参数给要触发的脚本
        $url_array = parse_url($url); //获取URL信息,以便平凑HTTP HEADER
        $port = isset($url_array['port'])? $url_array['port'] : 80; 
      
        $fp = fsockopen($url_array['host'], $port, $errno, $errstr, 30); 
        if (!$fp) …{
                return FALSE;
        }
        $getPath = $url_array['path'] ."?". $url_array['query'];
        if(!empty($post_data))…{
                $method = "POST";
        }
        $header = $method . " " . $getPath;
        $header .= " HTTP/1.1\r\n";
        $header .= "Host: ". $url_array['host'] . "\r\n "; //HTTP 1.1 Host域不能省略
        /**//*以下头信息域可以省略
        $header .= "User-Agent: Mozilla/5.0 (Windows; U; Windows NT 5.1; en-US; rv:1.8.1.13) Gecko/20080311 Firefox/2.0.0.13 \r\n";
        $header .= "Accept: text/xml,application/xml,application/xhtml+xml,text/html;q=0.9,text/plain;q=0.8,image/png,q=0.5 \r\n";
        $header .= "Accept-Language: en-us,en;q=0.5 ";
        $header .= "Accept-Encoding: gzip,deflate\r\n";
         */

        $header .= "Connection:Close\r\n";
        if(!empty($cookie))…{
                $_cookie = strval(NULL);
                foreach($cookie as $k => $v)…{
                        $_cookie .= $k."=".$v."; ";
                }
                $cookie_str =  "Cookie: " . base64_encode($_cookie) ." \r\n";//传递Cookie
                $header .= $cookie_str;
        }
        if(!empty($post_data))…{
                $_post = strval(NULL);
                foreach($post_data as $k => $v)…{
                        $_post .= $k."=".$v."&";
                }
                $post_str  = "Content-Type: application/x-www-form-urlencoded\r\n";//POST数据
                $post_str .= "Content-Length: ". strlen($_post) ." \r\n";//POST数据的长度
                $post_str .= $_post."\r\n\r\n "; //传递POST数据
                $header .= $post_str;
        }
        fwrite($fp, $header);
        //echo fread($fp, 1024); //我们不关心服务器返回
        fclose($fp);
        return true;
}

    現在,就可以透過這個函數來觸發一個PHP腳本的執行,然後函數就會回傳。 我們就可以接著執行下一步操作了。

   還有一個問題就是,當客戶端斷開連線以後。也就是triggerRequest發送請求後,立即關閉了連接,那麼可能會造成伺服器端正在執行的腳本退出。

   在PHP 內部,系統維護連接狀態,其狀態有三種可能的情況:

    * 0 – NORMAL(正常)

    * 1 – ABORTED(異常退出)

    * 2 – TIMEOUT(逾時)

     當PHP 腳本正常地執行NORMAL 狀態時,連線為有效。當客戶端中斷連線時,ABORTED 狀態的標記將會被開啟。遠端客戶端連線的中斷通常是由使用者點擊 STOP 按鈕所導致的。當連線時間超過 PHP 的時限(請參閱 set_time_limit() 函數)時,TIMEOUT 狀態的標記將會被開啟。

     可以決定腳本是否需要在客戶端中斷連線時退出。有時讓腳本完整地運行會帶來很多方便,即使沒有遠端瀏覽器接受腳本的輸出。預設的情況是當遠端客戶端連線 中斷時腳本將會退出。此處理過程可由 php.ini 的 ignore_user_abort 或由 Apache .conf 設定中對應的「php_value ignore_user_abort」以及 ignore_user_abort() 函數來控制。如果沒有告訴 PHP 忽略使用者的中斷,腳本將會中斷,除非透過 register_shutdown_function() 設定了關閉觸發函數。透過此關閉觸發函數,當遠端使用者點擊 STOP 按鈕後,腳本再次嘗試輸出資料時,PHP 將會偵測到連線已中斷,並呼叫關閉觸發函數。

     腳本也有可能被內建的腳本計時器中斷。預設的超時限制為 30 秒。這個值可以透過設定 php.ini 的 max_execution_time 或 Apache .conf 設定中對應的「php_value max_execution_time」參數或 set_time_limit() 函數來變更。當計數器逾時的時候,腳本將會類似於上述連線中斷的情況退出,先前被註冊過的關閉觸發函數也會在這時執行。在這個關閉觸發函數中,可以透過調 用 connection_status() 函數來檢查逾時是否導致關閉觸發函數被呼叫。如果超時導致了關閉觸發函數的調用,則該函數將返回 2。

     需要注意的一點是 ABORTED 和 TIMEOUT 狀態可以同時有效。這在告訴 PHP 忽略使用者的退出操作時是可能的。 PHP 將仍然注意用戶已經中斷了連線但腳本仍在運行的情況。如果到了運行的時間限制,腳本將被退出,設定過的關閉觸發函數也將被執行。在這時會發現函數 connection_status() 傳回 3。

      所以还在要触发的脚本中指明:

ignore_user_abort(TRUE); //如果客户端断开连接,不会引起脚本
abort.set_time_limit(0);//取消脚本执行延时上限

     或者,也可以使用:

register_shutdown_function(callback fuction[, parameters]);//注册脚本退出时执行的函数

以上是如何使用fscok實現非同步呼叫PHP的程式碼案例的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn