需求管理今天碰到一个问题,就是用户反映,添加备注等接口响应速度慢,6秒多,影响用户体验。
查找原因,发现是因为addComment接口里最后有发送邮件的操作,这个接口是同步的, 需要6秒左右返回。
addComment请求里,有三个主要操作,都跟网络环境息息相关:
** 主要操作**
1
连接数据库,修改数据;
2
请求日志接口,添加用户日志;
3
调用邮件服务的接口发送邮件;
这个三个操作是串行的顺序的执行的,都执行完才能返回,但是第三步耗时6秒,导致界面卡。这个邮件服务是别人的接口,需要他们去排查吧。
要解决这个问题,我这边需要把接口改成异步的,首先想到的是用多线程,把第三步发送邮件的操作放到工作线程里,这样当前线程可以及时返回结果给客户端,不能让用户等太长时间,因为他们是上帝。
问题来了,php自身不支持多线程,听说有个扩展-pthreads可以实现,那就安装吧。
pthreads的安装
pthreads扩展
Windows下PHP多线程扩展pthreads的安装
pthreads扩展安装步骤
1.查看phpinfo()
获取PHP版本号及位数(x86表示32位,x64表示64位)、编译器版本、PHP配置文件加载所在位置等。如下图所示:
2.pthreads扩展
下载地址:
http://windows.php.net/downloads/pecl/releases/pthreads/
3.在扩展列表中找到对应版本
4.php_pthreads-2.0.9-5.5-ts-vc11-x86.zip参数详解
2.0.9代表pthreads的版本号5.5代表php的版本号ts表示php是线程安全版本vc11表示php要MSVC11 (Visual C++ 2012)编译器编译x86则表示PHP版本是32位5.解压缩包
复制php_pthreads.dll到D:\xampp\php\ext目录下复制pthreadVC2.dll到D:\xampp\php\目录下复制pthreadVC2.dll到D:\xampp\apache\bin目录下复制pthreadVC2.dll到C:\windows\system32目录下7.重启xampp服务器
重启服务器后,查看phpinfo()就能看到pthreads扩展就表示安装成功,如下图:
Note:
安装时看好版本:
php5需要安装pthreads v2版,
php7可以安装pthreads v3
done-用官方例子测试
test_.php:
然后在项目里尝试,把发送邮件的代码放到子线程里,日志里也显示调用成功了,但是就是没收到邮件。费解,怀疑是子线程里的操作没完成, 主线程返回,php进程退出了?导致邮件发送失败?
但是,发送邮件的接口返回的是成功了,没有什么错误,而且,在接口最后,加上sleep(10)也不行,很奇怪。
过程中,发现一个通过curl并发请求的开源项目 ParallelCurl ,但并不能解决我们的问题。只能另想办法。
最后,利用php的系统调用,开启新的进程来实现。
php 提供了fsockopen函数,此函数的功能为初始化一个套接字连接到指定主机,默认情况下将以阻塞模式开启套接字连接。当然你可以通过stream_set_blocking()将它转换到非阻塞模式。
这是关键。所以,思路就是:开启一个非阻塞的套接字连接到本机,本机收到之后作一些耗时处理。
现在要做的就是: 把第3步耗时的邮件发送操作放到其他进程里执行。
这个接口完成第2步后, 与服务器建立非阻塞的socket连接,通过http协议告诉服务器:帮忙发个邮件啊,东西都放这了,我还有事,先走了,拜拜 。
之后addComment这个接口也就可以马上给客户端回复。
服务端另外一个接口收到发送邮件请求后,会异步的完成邮件发送。
废话不多说,直接上代码:
addComment接口伪代码:
发送邮件的代码:
这样就完成了异步操作。
改完后的效果:
本文转载自 360质量效能