天天看点

PHP异步的玩法

php是世界上最好的语言,但是总被“同行们”吐槽不支持异步。其实我们要实现异步也非常简单,之前看到鸟哥的一篇写php异步执行的博文 php实现异步调用方法研究,这篇文章还是08年的,到今天php发展快10年了,对于异步调用也有了更多新的玩法。

一是通过渲染前端页面,使用js执行ajax,这种方式现在还适用。只是受限于业务场景,因为只能在浏览器中调用,遇到接口请求就不行了。

二是通过popen()方法打开一个指向进程的管道,每个请求会多起一个进程。忽略进程来看最主要的原因是数据的传输特别不方便,使用场景有限。

三是使用curl扩展,通过设置timeout超时参数,能实现离弦之箭的效果。不过这种方法会主动断开连接。被调用的服务如果有做连接检测,也会中断服务端脚本的执行。比如我们请求 微信的某个费时接口(20s),我们调用1s就断开连接,微信端是否会维持请求执行20s是不可控的。所以这种方法不推荐大家使用。

四方法与curl类似,通过fsockopen创建socket连接访问远程服务,不循环获取请求结果。一样会有三中连接被断开的问题。

curl扩展已支持毫秒配置,将 curlopt_timeout 改为 curlopt_timeout_ms 即可生效(curl 版本 >= libcurl/7.21.0,老服务器要检查版本),但还是我前面说的需要服务端配合,不然接口的调用结果不可控。

curl扩展已支持并发,我们能一次访问n个接口,执行时间取最长接口的时间。比如我们能一次访问 京东支付(1s),微信支付(1.2s),支付宝(0.8s)不同服务的三个接口,总耗时才1.2s。详细用法 curl_multi_init

类似node.js的异步io框架swoole,能很好的实现异步调用;不过swoole理论上不能算php框架,他算是php功能的扩展。所以除非项目都用swoole写,不然也是享受不到异步io的福利。

对yield的支持,能实现调度器的功能,写单进程的服务时能大展拳脚,特别是实现协程,异步更不在话下。不过在多进程的web服务上没有太大的使用场景,看未来会不会有新的玩法吧。

当然还有很多新的特性,这里不再细说,总之php越是被黑越是能快速发展。

我给这个功能取了一个很生动的名字--离弦之箭。代表异步调用,我们的弓箭射出去后并不关心它的结果因为发送这个动作做了就行。比如打个10m的log,通知10个人(发10条短信)。

代码说明:首先arrow类是个单例类,减少多次调用的开销。run()方法传递的是一个匿名函数,这样我们能非常方便的传递参数,并且保留上下文。

这个类最难的地方在于多进程的处理。因为我们要尽可能快的将数据返回给用户,所以主进程越快结束越好。但是我们又需要子进程来执行我们耗时的操作,执行完退出才行。如果不等子进程执行完就将父进程退出会出现什么结果呢?结果就是子进程会常驻内存变成僵死进程。那我们有什么办法让子进程执行完之后就自动结束呢?答案是很难……那么儿子进程这么不听话,孙子进程会不会听话一点呢??答案是孙子进程执行结束后会被系统进程回收并销毁(还是孙子听话)。所以我在代码中使用了如下方法:当前请求进程fork出子进程,子进程fork出孙子进程,主进程和子进程都先行退出,最后由孙子进程来执行耗时操作,最后完美的解决了僵死进程问题。