PHP开启异步多线程执行脚本

场景要求

客户端调用服务器a.php接口,需要执行一个长达5s-20s不等的耗资源操作,但是客户端响应请求时间为5秒(微信公众账号服务器请求响应超时时间),5s以上无回复即断开连接。

解决设想

客户端调用a.php之后,a.php执行异步多线程操作调用b.php,a.php调用成功后即刻反馈给客户端回执,b.php自动执行耗资源操作。

PHP没有真正实现多线程操作的方法。所以需要通过其它手段来进行模拟多线程。

方案一

利用CURL非阻塞调用b.php,实现过程可以参考http://blog.csdn.net/linvo/article/details/5956629

但是有一个问题,就是a.php会继续等待b.php的响应。

于是临时想了一个解决方案:

在此处代码中,将$curlopt_timeout改为1

  1. /**
  2. * 单个CURL调用超时限制
  3. */
  4. public $curlopt_timeout = 1;
  5. private $param = array();

但是这样做就违背了curl本身的逻辑限制。

方案二

利用socket在a.php中加入以下代码

  1. $fp = fsockopen("test.com", 80, $errno, $errstr, 30);
  2. if (!$fp){
  3. echo ‘error fsockopen‘;
  4. }
  5. else{
  6. stream_set_blocking($fp,0);
  7. $http = "GET /test/b.php HTTP/1.1\r\n";
  8. $http .= "Host: test.com\r\n";
  9. $http .= "Connection: Close\r\n\r\n";
  10. fwrite($fp,$http);
  11. fclose($fp);
  12. }

即可实现a.php调用b.php无阻塞。

代码中stream_set_blocking函数用来设定socket链接为无阻塞方式(默认为阻塞)。

在使用方案二以后,遇到了一个问题,即客户端短时间内多次调用a.php,出现部分请求 没有执行b.php 的情况。

解决方法:在Nginx的nginx.conf文件中,查看worker_processes为1,判断服务端响应请求的线程启动限制太大,得知服务器本身配置为双核CPU,判断2-4线程比较合适,于是修改worker_processes为4.问题得到解决!

原文地址:https://www.cnblogs.com/hjcan/p/10436435.html

时间: 2024-08-27 15:28:31

PHP开启异步多线程执行脚本的相关文章

同步程序与异步程序执行原理

JavaScript中计算机程序分为同步执行与异步执行.同步执行:就是正常的计算机程序的执行, 顺序控制语句:从上到下,从左到右 循环控制语句:for,while,do...while,for...in,forEah() 分支控制语句:if,switch异步执行:是一种特殊的程序执行程序.setInterval,setTimeout,事件的绑定:onclick.ajex请求.所谓的异步程序的执行: 1:所有的异步程序都是在同步程序结束之后再执行. 2:异步程序的执行,如果时间相同,看代码的先后顺

多线程异步执行脚本

获取远程计算机信息: $d=get-date $servers="10.4.34.41","10.4.34.40","10.4.34.36","10.4.34.37","10.4.34.39" $serverpass="Dell1950" $UserName="Administrator" #定义线程数量$throttleLimit = 5$SessionState

高性能网站优化-确保异步加载脚本时保持执行顺序

<高性能网站建设进阶指南> 脚本如果按照常规方式加载,不仅会阻塞页面中其他内容的下载,还会阻塞脚本后面所有元素的渲染.异步加载脚本可以避免这种阻塞现象,从而提高页面加载速度.但是性能的提升是要付出代价的.代码的异步执行可能会出现竞争状态.简单地说就是页面内部的脚本需要的标示符如果是在外部文件中定义的,而当外部文件异步加载的时候,如果没有保证外部文件和内部脚本执行顺序,很有可能会出现未定义标示符的错误 当异步加载的外部脚本与行内脚本之间存在代码依赖时,就需要通过一种保证执行顺序的方法来整合这两个

异步执行脚本

异步执行脚本 script 元素的 async 属性可以使关联脚本相对于页面的其余部分进行异步加载和执行. 也就是说,在继续对页面进行解析的同时,在后台加载并执行脚本.如果页面使用了要占用大量处理时间的脚本,那么使用 async 属性可以显著提高页面加载性能. async 属性. async 属性是万维网联合会 (W3C)  HTML5 规范的一部分,它是为以下情形设计的:对于脚本不存在依赖关系,但是脚本仍需要尽快运行.在以下假设示例中,如果不使用 async(或 defer)属性,则脚本可能会

.NET 异步多线程,Thread,ThreadPool,Task,Parallel

今天记录一下异步多线程的进阶历史,以及简单的使用方法 主要还是以Task,Parallel为主,毕竟用的比较多的现在就是这些了,再往前去的,除非是老项目,不然真的应该是挺少了,大概有个概念,就当了解一下进化史了 1:委托异步多线程,所有的异步都是基于委托来实现的 #region 委托异步多线程 { //委托异步多线程 Stopwatch watch = new Stopwatch(); watch.Start(); Console.WriteLine($"开始执行了,{DateTime.Now.

java多线程执行问题

class Demo extends Thread{ public Demo(String name){ super(name); } public void run(){ for(int i=0; i<6; ++i){ System.out.println("i = " + i + "......Thread=" + Thread.currentThread().getName()); try{ Thread.sleep(100); }catch(Inter

Linux下设置定期执行脚本

下面针对的是非ubuntu环境,会在文章末尾介绍ubuntu的一些区别. 在Linux下,经常需要定期的执行一些脚本从而来实现一些功能. 在Linux下我们用crontab来实现定期的执行脚本这个功能,下面就介绍一下crontab的使用.以及我遇到的一些问题 一. crontab的使用说明 1. crond 是linux用来定期执行程序的命令.当安装完成操作系统之后,默认便会启动此任务调度命令.crond命令每分钟会定期检查是否有要执行的工作,如果有要执行的工作便会自动执行该工作.而linux任

通过SqlClr制作Sql自动化批量执行脚本

通过SqlClr制作Sql自动化批量执行脚本 在与同事一起做项目时,看到同事用sqlclr做批量执行脚本,感觉挺新奇的就上网搜集资料自己模仿跟做了个案例, 感觉挺不错的,现在想和大家分享一下,可能存在些错误的地方,大家就做个小参考吧.... 1.我们在做数据迁移或是数据库结构修改时,通常会写一些脚本文件之后逐个运行.但是如果有数十或数百个脚本文件, 那么就可以通过SqlClr制作Sql自动化执 2.比如现在ImportDataScript文件夹内有些脚本文件: 3.我们想让这9个脚本文件自动的依

批处理实现多线程执行命令列表文件

批处理实现多线程执行命令列表 工作中碰到多线程执行命令列表的情况,研究一番,编写出来与大家分享.高手看了也请指点,指教一些更简单的办法. 批处理是一种单线程的简单脚本,只有上条命令执行完后,才能执行下条命令.如果上条命令执行花费很长时间,如超时,连接失败不断尝试等,下一条命令头发白了,可能还等不到它执行. 百度问答上找到一个方法可以实现多线程,原理:利用bat调用bat来实现多线程. 例如:网管希望同时ping局域中所有主机,实现方法如下. 首先,建立两个批处理文件: 1.bat文件代码如下: