浅谈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 error ($curl_errno): $curl_error\n"; } else { echo "data received: $data\n"; } } else { // server sleep(10); echo "done."; } ?>
其他一些技巧:
1. 按照经验总结是:curl 版本 >= libcurl/7.21.0 版本,毫秒级超时是一定生效的,切记。
2. curl_multi的毫秒级超时也有问题。。单次访问是支持ms级超时的,curl_multi并行调多个会不准
[流处理方式访问http]
除了curl,我们还经常自己使用fsockopen、或者是file操作函数来进行http协议的处理,所以,我们对这块的超时处理也是必须的。
一般连接超时可以直接设置,但是流读取超时需要单独处理。
自己写代码处理:
$tmcurrent = gettimeofday(); $intusgone = ($tmcurrent['sec'] - $tmstart['sec']) * 1000000 + ($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处理后端&接口访问超时的解决方法全部内容了,希望大家多多支持~