好得很程序员自学网

<tfoot draggable='sEl'></tfoot>

PHP中实现异步调用多线程程序代码 - php高级应用

PHP中实现异步调用多线程程序代码

本文章详细的介绍了关于PHP中实现异步调用多线程方法,下面我们以给1000个用户发送一封推荐邮件,用户输入或者导入邮件账号了提交服务器执行发送来讲述.

比如现在有一个场景,给1000个用户发送一封推荐邮件,用户输入或者导入邮件账号了提交服务器执行发送,代码如下:

<?php    $sqlservercount = count ( $emailarr );    for ( $i =0; $i < $count ; $i  )    {      sendmail(.....); //发送邮件     }    ?> 

这段代码用户体验极差,也无法实际运用,首先发送这么多邮件会产生服务器运行超时,其实漫长的用户等待时间会让用户对系统产品怀疑和失去信心,但是用户不需要等待到1000封邮件都发送完毕了才提交发送成功,我们完全可以提交后台后直接给用户提示发送成功,然后让后台程序静默依次发送.

这个时候我们就需要[异步执行]技术来执行代码,异步执行的特点是后台静默执行,用户无需等待代码的执行结果,使用异步执行的好处.

1.摆脱了应用程序对单个任务的依赖性

2.提高了程序的执行效率

3.提高了程序的扩展性

4.在一定场景提高了用户体验

5.因为PHP不支持多线程,使用异步调用的请求多个HTTP的方式达到了程序并行执行效果,但是注意的是请求的HTTP过多的话,会大大加大了系统的开销.

用户体验:用户等待->发送完毕

朋友们就会问,怎么缺少发信环节?

OK,发信环节就在用户提交请求的时候,把发信任务转给了一个单独处理发信的php程序处理了,当用户看见[发送完毕]的时候其实信还没发送完,这个时候,发信程序正在后台努力的工作着,一封一封的向外发送.

sendmail.php代码如下:

<?php  $domain = "www.***.com" ;  $url = "/system_mail.php" ;  $par = "email=" .implode( ',' , $emailarr ). "&........" ;  $header  =  "POST $url HTTP/1.0rn" ;  $header  .=  "Content-Type: application/x-www-form-urlencodedrn" ;  $header  .=  "Content-Length: "  .  strlen ( $par ) .  "rnrn" ;  $fp  = @ fsockopen  ( $domain , 80,  $errno ,  $errstr , 30);  fputs  ( $fp ,  $header  .  $par );  fclose( $fp );    echo   '' 发送完毕';  ?> 

system_mail.php

<?php  ini_set ( "ignore_user_abort" ,true);  ignore_user_abort(true); //此处的代码需要php.ini开启相关的选项,保证php执行不超时的,不明白,参考我的另一篇文章 [关闭浏览器后,php脚本会不会继续运行]   //获取email地址,发信,此处为发信代码   ?> 

好了,改成异步方式后,用户提交信息,可以立即得到结果[发送完毕],信呢,会在后台一封一封的发送,直到发送完毕.

经过试验,总结出来几种方法,和大家share.

1. 最简单的办法,就是在返回给客户端的HTML代码中,嵌入AJAX调用,或者,嵌入一个img标签,src指向要执行的耗时脚本.

这种方法最简单,也最快,服务器端不用做任何的调用.

但是缺点是,一般来说Ajax都应该在onLoad以后触发,也就是说,用户点开页面后,就关闭,那就不会触发我们的后台脚本了.

而使用img标签的话,这种方式不能称为严格意义上的异步执行,用户浏览器会长时间等待php脚本的执行完成,也就是用户浏览器的状态栏一直显示还在load.

当然,还可以使用其他的类似原理的方法,比如script标签等等.

2. popen()

resource popen ( string command, string mode );

打开一个指向进程的管道,该进程由派生给定的 command 命令执行而产生,打开一个指向进程的管道,该进程由派生给定的 command 命令执行而产生.

所以可以通过调用它,但忽略它的输出,代码如下:

pclose(popen("/home/xinchen/backend.php &", 'r')); 

这个方法避免了第一个方法的缺点,并且也很快,但是问题是,这种方法不能通过HTTP协议请求另外的一个WebService,只能执行本地的脚本文件,并且只能单向打开,无法穿大量参数给被调用脚本。

并且如果,访问量很高的时候,会产生大量的进程,如果使用到了外部资源,还要自己考虑竞争.

3. 使用CURL

这个方法,设置CUROPT_TIMEOUT为1(最小为1,郁闷)。也就是说,客户端至少必须等待1秒钟,代码如下:

$ch  = curl_init();    $curl_opt  =  array (CURLOPT_URL,  'http://www.example.com/backend.php' ,   CURLOPT_RETURNTRANSFER, 1,  CURLOPT_TIMEOUT, 1,);    curl_setopt_array( $ch ,  $curl_opt );  curl_exec( $ch );  curl_close( $ch ); 

4. 使用fsockopen

这个方法应该是最完美的,但是缺点是,你需要自己拼出HTTP的header部分,代码如下:

$fp  =  fsockopen ( "www.example.com" , 80,  $errno ,  $errstr , 30);    if  (! $fp ) {         echo   "$errstr ($errno)<br />n" ;    }  else  {         $out  =  "GET /backend.php / HTTP/1.1rn" ;         $out  .=  "Host: www.example.comrn" ;         $out  .=  "Connection: Closernrn" ;            fwrite( $fp ,  $out );         /*忽略执行结果     while (!feof($fp)) {     echo fgets($fp, 128);   //开源代码phpfensi.com   }*/         fclose( $fp );    } 

查看更多关于PHP中实现异步调用多线程程序代码 - php高级应用的详细内容...

  阅读:42次