PHP中几种常见的超时处理全面总结(二)

【PHP本身超时处理】
[ PHP-fpm ]
配置:php-fpm.conf

代码如下:

<configuration> 
//... 
    Setsthelimitonthenumberofsimultaneousrequeststhatwillbeserved. 
    EquivalenttoApacheMaxClientsdirective. 
    EquivalenttoPHP_FCGI_CHILDRENenvironmentinoriginalphp.fcgi 
    Usedwithanypm_style. 
    #php-cgi的进程数量 
    <value name="max_children">128</value> 
        Thetimeout(inseconds)forservingasinglerequestafterwhichtheworkerprocesswillbeterminated 
        Shouldbeusedwhen'max_execution_time'inioptiondoesnotstopscriptexecutionforsomereason 
        '0s'means'off' 
        #php-fpm 请求执行超时时间,0s为永不超时,否则设置一个 Ns 为超时的秒数 

        <valuename="request_terminate_timeout">0s</value> 
        Thetimeout(inseconds)forservingofsinglerequestafterwhichaphpbacktracewillbedumpedtoslow.logfile 
        '0s'means'off' 
    <valuename="request_slowlog_timeout">0s</value> 
</configuration> 

说明:
在php.ini中,有一个参数max_execution_time可以设置PHP脚本的最大执行时间,但是,在php-cgi(php-fpm)中,该参数不会起效。
真正能够控制PHP脚本最大执行时:
request_terminate_timeout=0s
就是说如果是使用mod_php5.so的模式运行max_execution_time是会生效的,但是如果是php-fpm模式中运行时不生效的。

[ PHP ]
配置:php.ini
选项:
max_execution_time=30
或者在代码里设置:
ini_set(“max_execution_time”,30);
set_time_limit(30);
说明:
对当前会话生效,比如设置0一直不超时,但是如果php的safe_mode打开了,这些设置都会不生效。
效果一样,但是具体内容需要参考php-fpm部分内容,如果php-fpm中设置了request_terminate_timeout的话,那么max_execution_time就不生效。
【后端&接口访问超时】

【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允许执行的最长毫秒数。(在cURL7.16.2中被加入。从PHP5.2.3起可使用。) 
    CURLOPT_CONNECTTIMEOUT在发起连接前等待的时间,如果设置为0,则无限等待。 
    CURLOPT_CONNECTTIMEOUT_MS尝试连接等待的时间,以毫秒为单位。如果设置为0,则无限等待。在cURL7.16.2中被加入。从PHP5.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);//超时毫秒,cURL7.16.2中被加入。从PHP5.2.3起可使用 
        $data=curl_exec($ch); 
        $curl_errno=curl_errno($ch); 
        $curl_error=curl_error($ch); 
        curl_close($ch); 
        if($curl_errno>0){ 
            echo"cURLError($curl_errno):$curl_errorn"; 
        }else{ 
            echo"Datareceived:$datan"; 
        } 
    }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){ 
    returnfalse; 
} 

或者使用内置流处理函数stream_set_timeout()和stream_get_meta_data()处理:
代码如下:

<?php 
    //Timeoutinseconds 
    $timeout=5; 
    $fp=fsockopen("example.com",80,$errno,$errstr,$timeout); 
    if($fp){ 
        fwrite($fp,"GET/HTTP/1.0rn"); 
        fwrite($fp,"Host:example.comrn"); 
        fwrite($fp,"Connection:Closernrn"); 
        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"ConnectionTimedOut!"; 
    }else{ 
        echo$data; 
    } 
} 

file_get_contents超时:

代码如下:

$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); 
} 

【MySQL】
php中的mysql客户端都没有设置超时的选项,mysqli和mysql都没有,但是libmysql是提供超时选项的,只是我们在php中隐藏了而已。
那么如何在PHP中使用这个操作捏,就需要我们自己定义一些MySQL操作常量,主要涉及的常量有:
MYSQL_OPT_READ_TIMEOUT=11;
MYSQL_OPT_WRITE_TIMEOUT=12;
这两个,定义以后,可以使用options设置相应的值。
不过有个注意点,mysql内部实现:
1.超时设置单位为秒,最少配置1秒
2.但mysql底层的read会重试两次,所以实际会是3秒
重试两次+ 自身一次=3倍超时时间,那么就是说最少超时时间是3秒,不会低于这个值,对于大部分应用来说可以接受,但是对于小部分应用需要优化。
查看一个设置访问mysql超时的php实例:
代码如下:

//自己定义读写超时常量 (PHP代码)
    if(!defined('MYSQL_OPT_READ_TIMEOUT')){ 
        define('MYSQL_OPT_READ_TIMEOUT',11); 
    } 
    if(!defined('MYSQL_OPT_WRITE_TIMEOUT')){ 
        define('MYSQL_OPT_WRITE_TIMEOUT',12); 
    } 
    //设置超时 
    $mysqli=mysqli_init(); 
    $mysqli->options(MYSQL_OPT_READ_TIMEOUT,3); 
    $mysqli->options(MYSQL_OPT_WRITE_TIMEOUT,1); 
    //连接数据库 
    $mysqli->real_connect("localhost","root","root","test"); 
    if(mysqli_connect_errno()){ 
        printf("Connectfailed:%s/n",mysqli_connect_error()); 
    exit(); 
    } 
    //执行查询sleep1秒不超时 
    printf("Hostinformation:%s/n",$mysqli->host_info); 
    if(!($res=$mysqli->query('selectsleep(1)'))){ 
        echo"query1error:".$mysqli->error."/n"; 
    }else{ 
        echo"Query1:querysuccess/n"; 
    } 
    //执行查询sleep9秒会超时 
    if(!($res=$mysqli->query('selectsleep(9)'))){ 
        echo"query2error:".$mysqli->error."/n"; 
    }else{ 
        echo"Query2:querysuccess/n"; 
    } 
    $mysqli->close(); 
    echo"closemysqlconnection/n"; 

看后请多多点赞,感激不尽!!!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值