English | 简体中文 | 繁體中文 | Русский язык | Français | Español | Português | Deutsch | 日本語 | 한국어 | Italiano | بالعربية

PHPでバックエンド&インターフェースのアクセスタイムアウトを解決する方法について簡単に説明

【HTTPアクセス】

私たちがHTTPにアクセスする方法は多く、主に:curl、socket、file_get_contents()などの方法があります。

相手のサーバーが常に応答しない場合、私たちは悲劇的になります。サーバー全体をクラッシュさせる可能性があるので、HTTPアクセス時にはタイムアウトの問題も考慮する必要があります。

[CURLでHTTPアクセス]

CURLは私たちがよく使う、信頼性が高く、パフォーマンスが良く、いくつかの並行サポート機能があるHTTPプロトコルインターフェースのlibライブラリです。

CURL:

curl_setopt($ch, opt) 可以设置一些超时的设置,主要包括:

*(重要) CURLOPT_TIMEOUT 设置cURL允许执行的最长秒数。

*(重要) CURLOPT_TIMEOUT_MS 设置cURL允许执行的最长毫秒数。 (在cURL 7.16.2中被加わります。PHPから 5.2.3起可使用。 )

CURLOPT_CONNECTTIMEOUT 在发起连接前等待的时间,如果设置为0,则无限等待。

CURLOPT_CONNECTTIMEOUT_MS 尝试连接等待的时间,以毫秒为单位。如果设置为0,则无限等待。  在cURL 7.16.2中被加わります。PHPから 5.2.3开始可用。

CURLOPT_DNS_CACHE_TIMEOUT 设置在内存中保存DNS信息的时间,默认为120秒。

curl普通秒级超时:

$ch = curl_init();

curl_setopt($ch, CURLOPT_URL,$url);

curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);

curl_setopt($ch, CURLOPT_TIMEOUT, 60);   //只需要设置一个秒的数量就可以

curl_setopt($ch, CURLOPT_HTTPHEADER, $headers);

curl_setopt($ch, CURLOPT_USERAGENT, $defined_vars['HTTP_USER_AGENT']);

curl普通秒级超时使用:

curl_setopt($ch, CURLOPT_TIMEOUT, 60);

curl如果需要进行毫秒超时,需要增加:

curl_easy_setopt(curl, CURLOPT_NOSIGNAL, 1L);

或者是:

curl_setopt ( $ch,  CURLOPT_NOSIGNAL, true); 是可以支持毫秒级别超时设置的

curl一个毫秒级超时的例子:

<?php
if (!isset($_GET['foo'])) {
// Client
    $ch = curl_init('http://example.com/');
    curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
    curl_setopt($ch, CURLOPT_NOSIGNAL, 1);  
//注意,毫秒超时一定要设置这个
    curl_setopt($ch, CURLOPT_TIMEOUT_MS, 200); 
//タイムアウトミリ秒、cURL 7.16.2中被加わります。PHPから 5.2.3起で使用できます
    $data = curl_exec($ch);
    $curl_errno = curl_errno($ch);
    $curl_error = curl_error($ch);
    curl_close($ch);
    if ($curl_errno > 0) {
        echo "cURL エラー ($curl_errno): $curl_error\n";
    } else {
        echo "データを受信しました: $data\n";
    }
} else {
// サーバー
    sleep(10);
    echo "Done.";
}
?>

他の一些の技術:

1. 実務経験に基づいてまとめたのは:cURL バージョン >= libcurl/7.21.0バージョンでは、ミリ秒レベルのタイムアウトは常に有効ですので、忘れないでください。

2. curl_multiのミリ秒レベルのタイムアウトにも問題があります。。単一のアクセスではミリ秒レベルのタイムアウトがサポートされていますが、curl_multiで複数の並行アクセスを行うと正確ではありません

[ストリーム処理方式でHTTPにアクセス]

curl以外にも、私たちはよくfsockopenやfile操作関数を使用してHTTPプロトコルを処理しますので、この部分のタイムアウト処理も必須です。

一般的な接続タイムアウトは直接設定できますが、ストリームの読み取りタイムアウトは別々に処理する必要があります。

自作のコードで処理してください:


$intUSGone = ($tmCurrent['sec'] - $tmStart['sec']) * 10
+ ($tmCurrent['usec'] - $tmStart['usec']);
if ($intUSGone > $this->_intReadTimeoutUS) {
return false;
}

または、インラインストリーム処理関数 stream_set_timeout() と stream_get_meta_data() を使用して処理します:

<?php 
// Timeout in seconds 
$timeout = 5; 
$fp = fsockopen("example.com", 80, $errno, $errstr, $timeout); 
if ($fp) { 
    fwrite($fp, "GET"); / HTTP/1.0\r\n"); 
    fwrite($fp, "Host: example.com\r\n"); 
    fwrite($fp, "Connection: Close\r\n\r\n"); 
    stream_set_blocking($fp, true);  
//重要:ブロッキングモードに設定します
    stream_set_timeout($fp,$timeout);  
//タイムアウトの設定
    $info = stream_get_meta_data($fp); 
    while ((!feof($fp)) && (!$info['timed_out'])) { 
        $data .= fgets($fp, 4096); 
        $info = stream_get_meta_data($fp); 
        ob_flush; 
        flush(); 
    } 
    if ($info['timed_out']) { 
        echo "Connection Timed Out!"; 
    } else { 
        echo $data; 
    } 
}

file_get_contents 超時:

<?php
$timeout = array(
  'http' => array(
    'timeout' => 5 
//タイムアウト時間を設定します。単位は秒です
  )
);
$ctx = stream_context_create($timeout);
$text = file_get_contents("http://example.com/", 0, $ctx);
?>

fopen 超時:

<?php
$timeout = array(
  'http' => array(
    'timeout' => 5 
//タイムアウト時間を設定します。単位は秒です
  )
);
$ctx = stream_context_create($timeout);
if ($fp = fopen("http://example.com/", "r", false, $ctx)) {
 while( $c = fread($fp, 8192)) {
  echo $c;
 }
 fclose($fp);
}
?>

これで、編集者が皆さんに提供したPHPのバックエンドとインターフェースのタイムアウト解決方法の簡単な説明はすべて終わりました。皆さん、チュートリアルのサポートと応援をお願いします~

おすすめ